GreenDao 实现分库

前言

这是一篇介绍如何使用greenDao框架实现分库的文章。如果你对greenDao如何使用还不是很了解,下面有几篇文章推荐:
https://blog.csdn.net/qq_35956194/article/details/79167897
http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2017/0703/8146.html
https://blog.csdn.net/poorkick/article/details/55271660

解决问题

如果要使用greenDao实现分库,要解决下面两个问题:

  1. 如何实现在指定位置创建数据库
  2. 如何在指定数据库中创建对应的表

实现在指定位置创建数据库

通过对greenDao源码分析,知道DatabaseOpenHelper是继承SQLiteOpenHelper实现对数据库操作的。那么``是如何获取数据库路径的, 看下面部分源码:

private SQLiteDatabase getDatabaseLocked(boolean writable) {
   .... // 省略部分代码
    try {
        if (DEBUG_STRICT_READONLY && !writable) {
            final String path = mContext.getDatabasePath(mName).getPath();
            db = SQLiteDatabase.openDatabase(path, mFactory,
                    SQLiteDatabase.OPEN_READONLY, mErrorHandler);
        } else {
            db = mContext.openOrCreateDatabase(mName, mEnableWriteAheadLogging ?
                    Context.MODE_ENABLE_WRITE_AHEAD_LOGGING : 0,
                    mFactory, mErrorHandler);
        }
    } catch (SQLiteException ex) {
        if (writable) {
            throw ex;
        }
        Log.e(TAG, "Couldn't open " + mName
                + " for writing (will try read-only):", ex);
        final String path = mContext.getDatabasePath(mName).getPath();
        db = SQLiteDatabase.openDatabase(path, mFactory,
                SQLiteDatabase.OPEN_READONLY, mErrorHandler);
    }
    ....  // 省略部分代码
     
}

在上面代码中final String path = mContext.getDatabasePath(mName).getPath(); 这行代码就是获取数据库路径的。也就是通过Content 获取的数据库路径。如果在调用getDatabasePath(mName)方法时,把返回的路劲替换就可以了。思路是这样。

那么在什么地方可以干预getDatabasePath(mName)方法返回内容呢?最后发现Application 这个类继承 ContextWrapper, 如下:

public class Application extends ContextWrapper implements ComponentCallbacks2

那么就可以在应用的Application的继承实现类中重写getDatabasePath(mName) 方法, 如下:

public class App extends Application {

    @Override
    public File getDatabasePath(String name) {
        return super.getDatabasePath(name);
    }
}

只要在这个位置根据数据库名称返回相应位置就哦了。

其实不仅仅greenDao可以使用这种方式重写指定数据库位置,只要是通过SQLiteOpenHelper都可以。

如何在指定数据库中创建对应的表

下面的介绍都是基于一个示例介绍的, 示例结构为,两个数据库(db_adb_b)和四张表(TABLE_A_A,TABLE_A_B,TABLE_B_A,TABLE_B_B),

db_a 中有TABLE_A_A,TABLE_A_B
db_b 中有TABLE_B_A,TABLE_B_B

解决这个问题,要先从greenDao 生成的DaoMaster入手,看过DaoMater代码的都会知道, 创建表和删除就在这个类中的createAllTablesdropAllTables 静态方法进行。代码如下:

public static void createAllTables(Database db, boolean ifNotExists) {
    TableBaDao.createTable(db, ifNotExists);
    TableAaDao.createTable(db, ifNotExists);
    TableBbDao.createTable(db, ifNotExists);
    TableAbDao.createTable(db, ifNotExists);
}

public static void dropAllTables(Database db, boolean ifExists) {
    TableBaDao.dropTable(db, ifExists);
    TableAaDao.dropTable(db, ifExists);
    TableBbDao.dropTable(db, ifExists);
    TableAbDao.dropTable(db, ifExists);
}

如果使用这个DaoMaster中的这两个方法就会在每个库中都创建所这四张表,这不是想要的。
在往下看, 下面有一个静态内部抽象类OpenHelper, 源码如下:

public static abstract class OpenHelper extends DatabaseOpenHelper {
    public OpenHelper(Context context, String name) {
        super(context, name, SCHEMA_VERSION);
    }

    public OpenHelper(Context context, String name, CursorFactory factory) {
        super(context, name, factory, SCHEMA_VERSION);
    }

    @Override
    public void onCreate(Database db) {
        createAllTables(db, false);
    }
}

看到这里,在onCreate 中调用了createAllTables 方法在这个db下创建这个表。有了这些信息,就可以解决在不同的数据库中创建不同的表了。只要对每个数据库自定义一个master类,然后继承greenDao 生成的DaoMaster,重写createAllTablesdropAllTables方法, 在其中重写实现DatabaseOpenHelper, 在onCreate 中调用 createAllTables 创建对应的表。
说了这么多,可能已经晕了, 贴个示例代码:

public class DatabaseADaoMaster extends DaoMaster{
    public DatabaseADaoMaster(SQLiteDatabase db) {
        super(db);
    }

    public DatabaseADaoMaster(Database db) {
        super(db);
    }

    public static void createAllTables(Database db, boolean ifNotExists) {
        TableAaDao.createTable(db, ifNotExists);
        TableAbDao.createTable(db, ifNotExists);

    }

    public static void dropAllTables(Database db, boolean ifExists) {
        TableAaDao.dropTable(db, ifExists);
        TableAbDao.dropTable(db, ifExists);
    }

    public static class OpenHelper extends DatabaseOpenHelper {
        public OpenHelper(Context context, String name) {
            super(context, name, SCHEMA_VERSION);
        }

        public OpenHelper(Context context, String name, SQLiteDatabase.CursorFactory factory) {
            super(context, name, factory, SCHEMA_VERSION);
        }

        @Override
        public void onCreate(Database db) {
            createAllTables(db, false);
        }
    }
}

上面就greenDao分库实现的方法介绍,如有不好的地方请指出,一起学习一起进步。

下面使用greenDao实现分库功能,对部分内容进行封装的一个示例。

示例内容:

  1. 多库路径统一管理
  2. 统一管理不同库的Session
  3. 使用MigrationHelper升级数据库

示例项目地址:https://github.com/MrChao1991/GreenDaoMultDB

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,736评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,167评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,442评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,902评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,302评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,573评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,847评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,562评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,260评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,531评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,021评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,367评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,016评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,068评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,827评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,610评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,514评论 2 269

推荐阅读更多精彩内容