Android数据库Realm 升级,更换主键,更换字段类型

该篇文章主要讲解在android上使用Realm,关于升级的文章!
1.新增一个表(或者说新增一个类让其成为数据表)
2.更换已经存在的表中的字段类型(例如Int 类型更换为String类型)
3.更换已经存在的表中的主键

升级数据库很简单,在调用Realm实例的时候配置config时传入我们自己写好的MyMigration类即可,当然数据库的version也需要增加

object RealmHelper {
    private fun getRealmConfig(): RealmConfiguration {
        return RealmConfiguration.Builder()
                .name(RealmConstant.DB_NAME)
                .schemaVersion(RealmConstant.DB_VERSION)
                .migration(MyMigration())
                .build()
    }

    @JvmStatic
    fun getRealmInstance():Realm{
        return Realm.getInstance(getRealmConfig())
    }
}

接下来你要实现上述说的1,2,3只要在MyMigration类中实现即可

1.新增一个表

Realm数据库和传统SQL数据库增加表不一样,Realm只要增加一个表就要升级!新增表的类
例如我第一版本的DB_VERSION=0,现在我要新增一个表
a.DB_VERSION=1
b.实现新增的表类
c.在MyMigration处理升级

open class RedBookChapter(@PrimaryKey var chapterId: Int = -1, @Index var bookId: String = "") : RealmObject()

class MyMigration : RealmMigration {
    override fun migrate(realm: DynamicRealm, oldVersion: Long, newVersion: Long) {
        var oldV = oldVersion
        val schema = realm.schema
        if (oldV == 0L) {
            val bookChapterSchema = schema.create("RedBookChapter")
            bookChapterSchema?.let {
                it.addField("chapterId", Int::class.java, FieldAttribute.PRIMARY_KEY)
                        .addField("bookId", String::class.java, FieldAttribute.INDEXED)
                        .setRequired("bookId", true)
               
            }
           oldV++
        }
    }


    /**
     *  java.lang.IllegalArgumentException: Configurations cannot be different if used to open the same file.
     *  The most likely cause is that equals() and hashCode() are not overridden in the migration class:
     *  com.apusapps.reader.provider.realm.MyMigration
     */
    override fun hashCode(): Int {
        return MyMigration::class.java.hashCode()
    }

    override fun equals(other: Any?): Boolean {
        if (other == null) {
            return false
        }
        return other is MyMigration
    }
}

这样在每次操作数据库时,自然检查版本号就处理升级了!

2.更换已经存在的表中的字段的类型(例如Int 类型更换为String类型)

新增表中的字段很简单,addField即可,但是这里要说的是更换表中已经存在的字段的类型,并且字段名不变(即只更换字段的类型)
例如我第上一版本的DB_VERSION=1,现在我要更换字段的类型
a.DB_VERSION=2
b.处理更换字段的类
c.在MyMigration处理升级

原来的类
open class BookColl(@PrimaryKey var bookId: String = "",
                    var briefIntro: String? = "",
                    var majorCateId: Int? = -1,
                    var majorCateName: String? = "",
                    var minorCateId: Int? = -1,
                    var minorCateName: String? = ""
) : RealmObject()
修改后的类
open class BookColl(@PrimaryKey var bookId: String = "",
                    var briefIntro: String? = "",
                    var majorCateId: String? = "",
                    var majorCateName: String? = "",
                    var minorCateId: String? = "",
                    var minorCateName: String? = ""
) : RealmObject()
class MyMigration : RealmMigration {
    override fun migrate(realm: DynamicRealm, oldVersion: Long, newVersion: Long) {
        var oldV = oldVersion
        val schema = realm.schema
        if (oldV == 0L) {
            val bookChapterSchema = schema.create("RedBookChapter")
            bookChapterSchema?.let {
                it.addField("chapterId", Int::class.java, FieldAttribute.PRIMARY_KEY)
                        .addField("bookId", String::class.java, FieldAttribute.INDEXED)
                        .setRequired("bookId", true)
                
            }

            oldV++
        }

        if (oldV == 1L) {
            val bookCollSchema = schema.get("BookColl")
            bookCollSchema?.let {
                it.addField("majorCateId_temp", String::class.java)
                        .addField("minorCateId_temp", String::class.java)
                        .transform { obj ->
                            obj.setString("majorCateId_temp", obj.getInt("majorCateId").toString())
                            obj.setString("minorCateId_temp", obj.getInt("minorCateId").toString())
                        }
                        .removeField("majorCateId")
                        .removeField("minorCateId")
                        .renameField("majorCateId_temp", "majorCateId")
                        .renameField("minorCateId_temp", "minorCateId")
               
            }
            oldV++
        }
    }


    /**
     *  这里的hashCode,和equals同上面一样,这里省略
     */
   
}

说明下:

1.命名临时的字段majorCateId_temp,minorCateId_temp
2.将DB_VERSION=1中老用户的majorCateId和minorCateId这些字段迁移到DB_VERSION=2中的临时字段是上
3.移除老的字段
4.重新命名,将majorCateId_temp等字段改为之前的字段

3.更换已经存在的表中的主键

因为DB_VERSION=1时,新增了RedBookChapter,但是我的主键用错了,需要重新更换主键。
因为中间已经有了DB_VERSION=2了,所以这里DB_VERSION=3

a.DB_VERSION=3
b.处理更换主键的类
c.在MyMigration处理升级

更换主键之前的类
open class RedBookChapter(@PrimaryKey var chapterId: Int = -1, @Index var bookId: String = "") : RealmObject()
更换主键后的类
open class RedBookChapter(@PrimaryKey var hashCode:String = "",
                             var chapterId: Int = -1, @Index var bookId: String = "") : RealmObject()

注意这里的hasCode不是真正的hashCode,是我用bookId和chapterId拼装的:bookId.plus(chapterId.toInt())
要知道一个类的hashCode会随时变的,自己百度了解吧

class MyMigration : RealmMigration {
    override fun migrate(realm: DynamicRealm, oldVersion: Long, newVersion: Long) {
        var oldV = oldVersion
        val schema = realm.schema
        if (oldV == 0L) {
            val bookChapterSchema = schema.create("RedBookChapter")
            bookChapterSchema?.let {
                it.addField("chapterId", Int::class.java, FieldAttribute.PRIMARY_KEY)
                        .addField("bookId", String::class.java, FieldAttribute.INDEXED)
                        .setRequired("bookId", true)
                
            }

            oldV++
        }

        if (oldV == 1L) {
            val bookCollSchema = schema.get("BookColl")
            bookCollSchema?.let {
                it.addField("majorCateId_temp", String::class.java)
                        .addField("minorCateId_temp", String::class.java)
                        .transform { obj ->
                            obj.setString("majorCateId_temp", obj.getInt("majorCateId").toString())
                            obj.setString("minorCateId_temp", obj.getInt("minorCateId").toString())
                        }
                        .removeField("majorCateId")
                        .removeField("minorCateId")
                        .renameField("majorCateId_temp", "majorCateId")
                        .renameField("minorCateId_temp", "minorCateId")
               
            }
             oldV++
        }

        if (oldV == 2L) {
            val bookRedChapter = schema.get("RedBookChapter")
            bookRedChapter ?.let {
                it.addField("hashCode", String::class.java)
                        .addField("chapterId_temp",Int::class.java)
                        .transform { obj ->
                            obj.setString("hashCode", obj.getString("bookId").plus(obj.getInt("chapterId")))
                            obj.setInt("chapterId_temp", obj.getInt("chapterId"))
                        }
                        .removeField("chapterId")
                        .renameField("chapterId_temp","chapterId")
                        .addPrimaryKey("hashCode")
                        .setRequired("hashCode",true)
               
            }

           oldV++
        }
    }

    /**
     *  这里的hashCode,和equals同上面一样,这里省略
     */
}

到这里文章开头说的1,2,3要完成的事已经处理完毕了
其实只要处理好自己原来的类
处理好MyMigration即可

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