基于DBFlow实现数据表自动升级、多用户、多Module开发

DBFlow是Android开发中一个很优秀的ORM数据库框架,基于APT技术实现,拥有很好的性能,本文将会介绍DBFlow如何实现自动更新字段多用户数据库多Module开发

DBFlow托管地址:
https://github.com/Raizlabs/DBFlow

官方提供的更新字段方法

@Database(version = 2)
public class AppDatabase {
    @Migration(version = 2, database = AppDatabase.class)
    public class Migration2 extends AlterTableMigration<User> {
        public Migration2(Class<User> table) {
            super(table);
        }
        @Override
        public void onPreMigrate() {
            addColumn(SQLiteType.TEXT, "myColumn");
            addColumn(SQLiteType.REAL, "anotherColumn");
        }
    }
}

从上述的例子可以看到,每当我们修改字段的时候都必须手动编写升级的方法,做过字段升级的开发者应该都清楚,这种操作成本太高了,而且容易出错。

自动更新字段

下面介绍我在公司项目中如何改造DBFlow,实现自动更新字段。

1. 修改源代码DBFlow,重新打包

image.png

唯一的修改就是把@Column改成@Retention(RetentionPolicy.SOURCE),如果是RUNTIME类型,编译成class文件后,注解就消失了,因为我这个方案是通过@Column来实现字段的升级。

这是我自己修改的库
https://github.com/joyrun/DBFlow
引用方式:

compile "com.github.joyrun.DBFlow:dbflow-core:4.2.5"
compile "com.github.joyrun.DBFlow:dbflow:4.2.5"
apt "com.github.joyrun.DBFlow:dbflow-processor:4.2.5"
2. 实现自动升级

实现原理是,升级版本之后,检查数据库中所有的表的字段,和类中有有@Column的字段进行对比,如果不存在的就添加进来。

@Database(name = AppDatabase.NAME, version = AppDatabase.VERSION)
public class AppDatabase {
    //数据库名称
    public static final String NAME = "AppDatabase";
    //数据库版本号
    public static final int VERSION = 200;
    @Migration(version = AppDatabase.VERSION, database = AppDatabase.class)
    public static class DatabaseAutoUpdate extends BaseDatabaseAutoUpdate {
        @Override
        protected String getDatabaseName() {
            return AppDatabase.NAME;
        }
    }
    public static abstract class BaseDatabaseAutoUpdate extends BaseMigration {
        protected abstract String getDatabaseName();
        @Override
        public void migrate(DatabaseWrapper database) {
            try {
                List<Class<?>> classes = FlowManager.getDatabase(getDatabaseName()).getModelClasses();
                for (Class c : classes) {
                    try {
                        Cursor cursor = database.rawQuery("SELECT * FROM " + c.getSimpleName(), null);
                        Field[] fields = c.getDeclaredFields();
                        for (Field field : fields) {
                            if (field.isAnnotationPresent(Column.class)) {
                                if (cursor.getColumnIndex(field.getName()) < 0) {
                                    //缺少的字段
                                    QueryBuilder queryBuilder = new QueryBuilder().append("ALTER")
                                            .appendSpaceSeparated("TABLE")
                                            .appendSpaceSeparated(c.getSimpleName())
                                            .appendSpaceSeparated("ADD COLUMN")
                                            .appendSpaceSeparated(QueryBuilder.quoteIfNeeded(field.getName()));
                                    String sql = queryBuilder.getQuery();
                                    database.execSQL(sql);
                                }
                            }
                        }
                        cursor.close();
                    } catch (Exception e) {
                        RxJavaPluginUtils.handleException(e);
                    }
                }
            } catch (InvalidDBConfiguration e) {
                e.printStackTrace();
            }
        }
    }
}
3. 温馨提示

建议数据库版本号和App的versionCode保持一致,在Application编写检查代码,强提示,避免忘记修改数据版本号。

多用户数据库

比如类似微信,允许一个账号A注销后,登录另外一个账号B,注销后再登录账号A,账号A的数据库不会被清空。
如果要实现这种逻辑,我们就要针对不同的用户进行区分,不同的数据使用不同的数据库。

    public static void initDBFlow() {
        FlowManager.close();
        FlowConfig.Builder flowConfig = new FlowConfig.Builder(getContext()).openDatabasesOnInit(false);
        addDatabase(flowConfig,MarathonGeneratedDatabaseHolder.class,MarathonDatabase.class);
        addDatabase(flowConfig,AdvertGeneratedDatabaseHolder.class,AdvertDatabase.class);
        FlowManager.init(flowConfig.build());
    }
    private static void addDatabase(FlowConfig.Builder flowConfig,Class<? extends DatabaseHolder> databaseHolderClass,Class<? extends BaseDatabase> databaseClass){
        flowConfig.addDatabaseHolder(databaseHolderClass);
        flowConfig.addDatabaseConfig(new DatabaseConfig.Builder(databaseClass).extensionName(MyInfo.getInstance().getUid()+".db").build());
    }

DBFlow 4.x 提供DatabaseConfig给我们对数据进行配置,可以用来配置数据库名。我们可以通过这种方式来实现不同的用户使用不同的数据库。

温馨提示
  1. 切换账号的时候,必须再次调用以上方法,用于销毁DBFlow,重新初始化。

多Module开发

只要在Module的build.gradle增加配置,不同的Module需要配置不同的名称。

apt {
    arguments {
        targetModuleName 'SomeUniqueModuleName'
    }
}

就会生成对应的SomeUniqueModuleNameGeneratedDatabaseHolder类,初始化的时候添加进来即可

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

推荐阅读更多精彩内容