基于DBFlow實現(xiàn)數(shù)據表自動升級、多用戶、多Module開發(fā)

DBFlow是Android開發(fā)中一個很優(yōu)秀的ORM數(shù)據庫框架,基于APT技術實現(xiàn),擁有很好的性能,本文將會介紹DBFlow如何實現(xiàn)自動更新字段、多用戶數(shù)據庫、多Module開發(fā)。

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");
        }
    }
}

從上述的例子可以看到,每當我們修改字段的時候都必須手動編寫升級的方法,做過字段升級的開發(fā)者應該都清楚,這種操作成本太高了,而且容易出錯。

自動更新字段

下面介紹我在公司項目中如何改造DBFlow,實現(xiàn)自動更新字段。

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

image.png

唯一的修改就是把@Column改成@Retention(RetentionPolicy.SOURCE),如果是RUNTIME類型,編譯成class文件后,注解就消失了,因為我這個方案是通過@Column來實現(xiàn)字段的升級。

這是我自己修改的庫
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. 實現(xiàn)自動升級

實現(xiàn)原理是,升級版本之后,檢查數(shù)據庫中所有的表的字段,和類中有有@Column的字段進行對比,如果不存在的就添加進來。

@Database(name = AppDatabase.NAME, version = AppDatabase.VERSION)
public class AppDatabase {
    //數(shù)據庫名稱
    public static final String NAME = "AppDatabase";
    //數(shù)據庫版本號
    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. 溫馨提示

建議數(shù)據庫版本號和App的versionCode保持一致,在Application編寫檢查代碼,強提示,避免忘記修改數(shù)據版本號。

多用戶數(shù)據庫

比如類似微信,允許一個賬號A注銷后,登錄另外一個賬號B,注銷后再登錄賬號A,賬號A的數(shù)據庫不會被清空。
如果要實現(xiàn)這種邏輯,我們就要針對不同的用戶進行區(qū)分,不同的數(shù)據使用不同的數(shù)據庫。

    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給我們對數(shù)據進行配置,可以用來配置數(shù)據庫名。我們可以通過這種方式來實現(xiàn)不同的用戶使用不同的數(shù)據庫。

溫馨提示
  1. 切換賬號的時候,必須再次調用以上方法,用于銷毀DBFlow,重新初始化。

多Module開發(fā)

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

apt {
    arguments {
        targetModuleName 'SomeUniqueModuleName'
    }
}

就會生成對應的SomeUniqueModuleNameGeneratedDatabaseHolder類,初始化的時候添加進來即可

public void initialize(Context context) {
  FlowManager.init(FlowConfig.builder(context)
    .addDatabaseHolder(SomeUniqueModuleNameGeneratedDatabaseHolder.class)
    .build());
}
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

友情鏈接更多精彩內容