Android Realm数据库使用

当我们的app有数据需要保存到本地缓存时,可以使用file,sharedpreferences,还有sqlite。

sharedpreferences其实使用xml的方式,以键值对形式存储基本数据类型的数据。对于有复杂筛选查询的操作,file和sharedpreferences都不能满足了。sqlite可以满足有大量复杂查询要求的缓存数据操作。但是sqlite的使用略复杂,代码量很大,还好网上有很多优秀的orm框架可使用,比喻ORMlite,greenDao等。

ORMlite,greenDao这些框架都是在SQLite的基础上封装的ORM对象关系映射框架,简化了代码操作。

而今天的主角:Realm是一个可以替代SQLite以及ORM Libraries的轻量级数据库。

Realm官网上说了好多优点,我觉得选用Realm的最吸引人的优点有三点:

  1. 跨平台: 现在很多应用都是要兼顾iOS和Android两个平台同时开发。如果两个平台都能使用相同的数据库,那就不用考虑内部数据的架构不同,使用Realm提供的API,可以使数据持久化层在两个平台一无差异化的转换
  2. 简单易用: Core Data 和 SQLite 几余、繁杂的知识和代码足以下退绝大多数刚入门的开发者,而换用 Realm,则可以极大地学习成本,立即学会本地化存储的方法。毫不吹嘘的说,把官方最新文档完看一遍,就完全可以上手开发了
  3. 可视化: Realm 还提供了一个轻量级的数据库查看工具,在Mac Appstore 可以下载“Realm Browser”这个工具,开发者可以查看数据库当中的内容,执行简单的插入和删除数据的操作。毕竟,很多时,开发者使用数据库的理由是因为要提供一些所谓的“知识库”

相比SQLite,Realm更快并且具有很多现代数据库的特性,比如支持JSON,流式api,数据变更通知,以及加密支持,这些都为安卓开发者带来了方便。不多介绍,更详细的介绍参见官网:Realm Home | Realm.io

我们重点来说说Reaml的使用,看看到底爽在哪里。

环境配置:

1、在Project的build.gradle文件中添加依赖:

buildscript {repositories {mavenCentral()}dependencies {classpath "io.realm:realm-gradle-plugin:10.11.1"}
}

2、在app module的build.gradle文件的top添加下面代码:

apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-kapt'
apply plugin: 'realm-android'

配置完毕。

使用:

在整个使用的过程中,Realm是主角,我们先来看看Realm类中的一段翻译:

/**
     * Realm类可以对你的持久化对象进行存储和事务管理,可以用来创建RealmObjects实例。领域内的对象可以在任何时间查询和读取。
     * 创建,修改和删除等操作必须被包含在一个完整的事务里面,后面的代码会讲到。
     * 该事务确保多个实例(在多个线程)可以在一个一致的状态和保证事务在ACID前提下,访问相同的对象。
     *
     * 当一个Realm实例操作完成后,切记不要忘记调用close()方法。否则会导致本地资源无法释放,引起OOM。
     *
     * Realm实例不能在不同的线程间访问操作。确切的说,你必须在每个要使用的线程上打开一个实例
     * 每个线程都会使用引用计数来自动缓存Realm实例,所以只要引用计数不达到零,
     *  调用getInstance(RealmConfiguration)方法将会返回缓存的Realm实例,应该算是一个轻量级的操作。
     *
     * 对于UI线程来说,打开和关闭Realm实例,应当放在onCreate/onDestroy或者onStart/onStop方法中
     *
     *  在不同的线程间,Realm实例使用Handler机制来调整他的状态。也就是说,Realm实例在线程中,如果没有Looper,是不能收到更新通知的,
     *  除非手动调用waitForChange()方法
     *
     * 在安卓Activity领域工作的一个标准模式可以在下面看到
     * 在Android Activity中,Realm的标准工作模式如下:
     *
     *
     * public class RealmApplication extends Application {
     *
     *     \@Override
     *     public void onCreate() {
     *         super.onCreate();
     *
     *         // The Realm file will be located in package's "files" directory.
     *         RealmConfiguration realmConfig = new RealmConfiguration.Builder(this).build();
     *         Realm.setDefaultConfiguration(realmConfig);
     *     }
     * }
     *
     * public class RealmActivity extends Activity {
     *
     *   private Realm realm;
     *
     *   \@Override
     *   protected void onCreate(Bundle savedInstanceState) {
     *     super.onCreate(savedInstanceState);
     *     setContentView(R.layout.layout_main);
     *     realm = Realm.getDefaultInstance();
     *   }
     *
     *   \@Override
     *   protected void onDestroy() {
     *     super.onDestroy();
     *     realm.close();
     *   }
     * }
     *
     *
     * Realm支持String和byte字段长度高达16MB
     * 参考连接:
     * <a href="http://en.wikipedia.org/wiki/ACID">ACID</a>
     * <a href="https://github.com/realm/realm-java/tree/master/examples">Examples using Realm</a>
     *
     */

部分源码分析:

public final class Realm extends BaseRealm {
//默认的文件名,是啥?
public static final String DEFAULT_REALM_NAME = RealmConfiguration.DEFAULT_REALM_NAME;
 
 
//怎么这么熟悉呢?是RxJava?
@Override
    @OptionalAPI(dependencies = {"rx.Observable"})
    public Observable<Realm> asObservable() {
        return configuration.getRxFactory().from(this);
    }
 
//下面这些方法,应该都能顾名思义吧
 
public <E extends RealmModel> void createAllFromJson(Class<E> clazz, JSONArray json) {
}
 
 
    public <E extends RealmModel> void createOrUpdateAllFromJson(Class<E> clazz, JSONArray json) {
}
 
 
    public <E extends RealmModel> E createOrUpdateObjectFromJson(Class<E> clazz, JSONObject json) {
}
 
 
    public <E extends RealmModel> E createObject(Class<E> clazz) {
}
 
 
    public <E extends RealmModel> E copyToRealmOrUpdate(E object) {
}
 
 
    public void executeTransaction(Transaction transaction) {
}
 
 
    public void delete(Class<? extends RealmModel> clazz) {
}
}

 RealmConfiguration类的说明翻译:

/**
 * 一个RealmConfiguration对象,可用来设置特定的Realm实例
 * RealmConfiguration实例只能通过io.realm.RealmConfiguration.Builder类的build()方法来创建
 * 想使用默认的RealmConfiguration实例,请使用io.realm.Realm#getDefaultInstance()方法。
 * 如果想使用自己配置RealmConfiguration实例的Realm实例,需要调用Realm#setDefaultConfiguration(RealmConfiguration)
 * 
 * <p>
 * 可以用下面代码创建一个最简单配置的实例:
 * RealmConfiguration config = new RealmConfiguration.Builder(getContext()).build())
 * 这样创建的实例,具有一下属性:
 * 
 * <ul>
 * <li>Realm的默认文件名是"default.realm"</li>
 * <li>"default.realm"文件保存在"Context.getFilesDir()"目录中</li>
 * <li>它的schema版本号设置为0</li>
 * </ul>
 */


部分源码分析:

public final class RealmConfiguration {
//默认文件名
public static final String DEFAULT_REALM_NAME = "default.realm";
 
//Rx工厂
private final RxObservableFactory rxObservableFactory;
 
//弱引用
private final WeakReference<Context> contextWeakRef;
 
/**
     * 
     * 从Asset目录中返回Realm文件名,还可以保存在Asset中?
     * @return input stream to the asset file.
     * @throws IOException if copying the file fails.
     */
    InputStream getAssetFile() throws IOException {
        Context context = contextWeakRef.get();
        if (context != null) {
            return context.getAssets().open(assetFilePath);
        } else {
        }
    }
 
/**
         * 使用app自己内置硬盘目录来存储Realm file。不需要任何扩展访问权限。
         * 默认目录为:/data/data/<packagename>/files,这个路径能否修改取决于供应商的具体实现
         * 
         * @param 参数context请使用application的context.
         */
        public Builder(Context context) {
            if (context == null) {
                throw new IllegalArgumentException("A non-null Context must be provided");
            }
            RealmCore.loadLibrary(context);
            initializeBuilder(context.getFilesDir());
        }

通过上面的翻译说明和源码分析,应该几乎明白了Realm的原理和基本使用了吧。总结下面几点:

  1. Realm保存的结果其实是在一个文件里面,默认的文件名是"default.realm",在"Context.getFilesDir()"目录中,即:/data/data/<packagename>/files/default.realm。意思是,当你在应用管理里面给当前app"清除数据",realm数据库的数据会丢失。故我们需要把默认的数据文件放到asset目录中,当数据库初始化时再copy到"Context.getFilesDir()"下。
  2. 在创建RealmConfiguration对象时,可以通过.assetFile(this,"realm file path in assets")方法指定初始化的数据库文件。Realm会把制定路径下的xxx.realm文件copy到Context.getFilesDir()目录中,以替换默认创建的空数据库文件。
  3. 可以设置默认文件名,通过RealmConfiguration类进行配置。路径似乎改不了,需要看具体设备供应商的实现。
  4. Realm的实例需要在每次的具体操作中获取,可以看成是一个数据操作的sessin,用完后必须close关闭。打开和关闭Realm实例,应当放在onCreate/onDestroy或者onStart/onStop方法中。
  5. Realm中似乎有RxJava的影子,支持链式异步任务?
  6. Realm中有个各种增删改差的方法,还可以根据JSON的数据实例化一个RealmObject子类java bean。
  7. 重点:切记,Realm数据库的主键字段不是自动增长的,需要自己设置,做添加的时候如果不给id字段值,默认会为0。后面再添加会报错,说id为0的数据已经存在。尤其是批量添加的时候要注意,当心出现只添加了一条记录的悲剧。
  8. 数据自动更新。mRealm.addChangeListener(this);//当数据库的数据有变化时,系统回调此方法。

经过上面的分析和总结,其实已经很明了了。还有些需要注意的地方,在代码中讲解。

application代码:

public class MyApplication extends Application {private String realmName = "dk.realm";@Overridepublic void onCreate() {super.onCreate();RealmConfiguration realmConfig = new RealmConfiguration.Builder(this).name(realmName)//.assetFile(this,"realm file path in assets,will copy this file to Context.getFilesDir() replace an empty realm file").build();Realm.setDefaultConfiguration(realmConfig);}
}

java Bean:

public class TestUser extends RealmObject {@PrimaryKeyprivate int userId;//id,主键@Requiredprivate String userName;//用户姓名,必填字段private String userPwd;//密码private int userAge;//年龄private String userAddress;//住址private String userWork;//工作private String userSex;//性别//private RealmList<E> list;  集合
//...
}

BaseDao,简单封装,把基本的增删改功能提取:

public class BaseDao {private Realm realm;public BaseDao(Realm realm) {this.realm = realm;}/*** 添加(性能优于下面的saveOrUpdate()方法)** @param object* @return 保存或者修改是否成功*/public boolean insert(RealmObject object) {try {realm.beginTransaction();realm.insert(object);realm.commitTransaction();return true;} catch (Exception e) {e.printStackTrace();realm.cancelTransaction();return false;}}
/*** 添加(性能优于下面的saveOrUpdateBatch()方法)** @param list* @return 批量保存是否成功*/public boolean insert(List<? extends RealmObject> list) {try {realm.beginTransaction();realm.insert(list);realm.commitTransaction();return true;} catch (Exception e) {e.printStackTrace();realm.cancelTransaction();return false;}}
//...
}

UserDao extends BaseDao:

/*** 单条保存demo*/public boolean addOneTest() {boolean bl = false;try{realm.beginTransaction();//在数据库中创建一个对象,主键默认值为0TestUser user = realm.createObject(TestUser.class);//(类,主键)//更新数据库各自段的值user.setUserName("admin");//主键字段的值由0更新为55。而不是直接创建了一个id为55的对象user.setUserId(55);//...realm.commitTransaction();bl = true;}catch (Exception e){e.printStackTrace();realm.cancelTransaction();}/*try{realm.beginTransaction();TestUser user2 = new TestUser("hibrid", "120250", 26, "赣州", "贼", "男");//不给id,会被默认为0//user2.setUserId(102);TestUser userWithId = realm.copyToRealm(user2);realm.commitTransaction();bl = true;}catch (Exception e){e.printStackTrace();realm.cancelTransaction();}*/return bl;}//init datapublic boolean init() {/*** 此处要注意,方法最后调用的是添加或者修改的方法。* 如果list的数据都不给id,则第一条记录添加成功后的id为0,后面的都在此基础上修改。* 最后的效果是,数据库只有一条记录,id为0,其他字段被更新为了最后一个对象的数据*/List<TestUser> list = new ArrayList<>();list.add(new TestUser(0,"android", "123123", 20, "河南常德", "传菜员", "女"));list.add(new TestUser(1,"angel", "13588889988", 21, "云南西双版纳", "飞行员", "男"));list.add(new TestUser(2,"adidass", "110119", 28, "云南德克萨斯州", "海员", "男"));list.add(new TestUser(3,"hijack", "250250", 39, "加州电厂", "厨师", "女"));list.add(new TestUser(4,"hibrid", "120250", 26, "赣州", "贼", "男"));list.add(new TestUser(5,"admin", "123456", 20, "湖北汉城", "程序员", "女"));return saveOrUpdateBatch(list);}/*** 条件查询** @return 返回结果集合*/public RealmResults<TestUser> findByAnyParams(HashMap<Object, Object> params) {//realm.where(TestUser.class)//可跟查询条件//.or()                      或者//.beginsWith()              以xxx开头//.endsWith()                以xxx结尾//.greaterThan()             大于//.greaterThanOrEqualTo()    大于或等于//.lessThan()                小于//.lessThanOrEqualTo()       小于或等于//.equalTo()                 等于//.notEqualTo()              不等于//.findAll()                 查询所有//.average()                 平均值//.beginGroup()              开始分组//.endGroup()                结束分组//.between()                 在a和b之间//.contains()                包含xxx//.count()                   统计数量//.distinct()                去除重复//.findFirst()               返回结果集的第一行记录//.isNotEmpty()              非空串//.isEmpty()                 为空串//.isNotNull()               非空对象//.isNull()                  为空对象//.max()                     最大值//.maximumDate()             最大日期//.min()                     最小值//.minimumDate()             最小日期//.sum()                     求和return realm.where(TestUser.class).findAll();}


MainActivity代码:

@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mRealm = Realm.getDefaultInstance();userDao = new UserDao(mRealm);//.../*** 数据库数据更新监听*/mRealm.addChangeListener(this);}//...@Overridepublic void onChange(Realm element) {findAll();}@Overrideprotected void onDestroy() {userDao = null;mRealm.close();super.onDestroy();}

增删改的代码注意事务,其他的都简单。
 


Android Realm数据库使用总结及采坑记录

Realm使用注意事项

  • Realm默认运行在主线程,使用时须开启异步任务
  • Realm本身是单例类,可以多线程并发调用,但是RealmObject则不允许并发,每个RealmObject都绑定了一个TreadId,必须在创建该RealmObject的线程中使用它.
  • 在子线程查询出的数据无法在主线程使用,自己的方案是:子线程查询,置换为自己的Bean类,然后在主线程使用
  • 没有主键的realmObject无法进行update操作.所以如果要使用realm.copyToRealmOrUpdate(realmObject),那么这个realmObject必须设置primaryKey
  • 如果Realm关闭,所有查询得到的RealmObject都不能使用了,解决方案是复制一份数据到内存中。
  • 操作数据库必须在transaction中完成

常见问题

Object not managed by Realm, so it cannot be removed

Realm不支持直接通过deleteFromRealm删除Bean类,即使该Bean extends RealmObject,否则会报此异常

正确姿势:

根据指定字段,从数据库中查询到该Bean,然后再删除

/*** 从数据库中删除CollectBean* @param conType* @param relateId*/
public void deleteCollectBeanByTypeAndId(String conType,int relateId){Realm realm = RealmUtils.getInstance().mRealm;CollectBean bean = realm.where(CollectBean.class).equalTo(CollectBean.CON_TYPE, conType).equalTo(CollectBean.RELATE_ID,relateId).findFirst();realm.beginTransaction();bean.deleteFromRealm();realm.commitTransaction();
}
Realm accessed from incorrect thread

RealmObject自带线程保护功能,只能在创建它的线程中访问,在子线程中不能访问。
也就是说,如果你在主线程中new了一个RealmObject对象 user,那么在子线程中是访问不了user对象的。
要想在子线程中访问,必须先将user存入Ream中,然后在子线程中query出来。

更多介绍:Realm的常规使用与线程中的坑 - 简书

is not part of the schema for this Realm

详细异常信息: java.lang.IllegalArgumentException: UserBean is not part of the schema for this Realm

需要调整plugin中的顺序,如下:

apply plugin: 'com.android.application'
apply plugin: 'com.bugtags.library.plugin'
apply plugin: 'android-apt'
apply plugin: 'realm-android'
apply plugin: 'com.neenbedankt.android-apt'


{bean}has a primary key, use ‘createObject(Class, Object)’ instead

详细异常信息: io.realm.exceptions.RealmException: ‘UserBean’ has a primary key, use ‘createObject(Class, Object)’ instead.

如果实体中已经通过@PrimaryKey标明了主键,那么想要通过createObject(Class<E>, Object)创建实体对象,则必须传入primaryKeyValue(主键值)

异步查询之坑
1.官方文档介绍 主线程操作Realm会卡顿/阻塞线程

官方表示Realm运行速度很快,足以在主线程运行,而后又表示其实还是会阻塞线程导致偶发的ANR,因此建议在子线程操作Realm.

2.子线程查询的数据,无法在主线程使用

解决方案:

子线程查询,置换为自己的Bean类,然后在主线程使用.

Realm.getDefaultInstance().executeTransactionAsync(new Realm.Transaction() {@Overridepublic void execute(Realm realm) {Person ziPerson = realm.where(Person.class).findFirst();personInfo = new PersonInfo();personInfo.setName(ziPerson.getName());personInfo.setAge(ziPerson.getAge());//Log 输出#Execute ] false..Person{name='小明', age=18}KLog.i((Looper.getMainLooper()==Looper.myLooper())+".."+ personInfo.toString());}
}, new Realm.Transaction.OnSuccess() {@Overridepublic void onSuccess() {//Log 输出#OnSuccess ] true..personInfo:Person{name='小明', age=18}KLog.i((Looper.getMainLooper()==Looper.myLooper())+".."+ personInfo.toString());}
}, new Realm.Transaction.OnError() {@Overridepublic void onError(Throwable error) {KLog.i(error.toString());}
});
RejectedExecutionException

详细异常信息:java.util.concurrent.RejectedExecutionException: Task java.util.concurrent.FutureTask@4dffbdd rejected from io.realm.internal.async.RealmThreadPoolExecutor@c09c352[Running, pool size = 17, active threads = 2, queued tasks = 100, completed tasks = 110]

解决方案:

不要在for循环中使用Realm,将数据存入集合中,然后开启事务,直接使用copyToRealmOrUpdate(realmObjectList)存储即可.

事务嵌套报异常

详细异常信息:The Realm is already in a write transaction in /Users/blakemeike/Working/release/realm/realm-library/src/main/cpp/io_realm_internal_SharedRealm.cpp line 116

原因 : 在一个事务中开启了另外一个事务.应避免这种情况.
 


对Android Realm数据库进行加密及版本升级管理

Realm从设计之初便是为适应移动端的使用场景。使用简洁,操作速度快。是一款很不错的移动端嵌入式数据库。
 

public class RealmDBConfig {private final static char[] hexArray = "0123456789ABCDEF".toCharArray();static RealmConfiguration realmConfig = null;private static int version = 1;  // 数据库版本号/*** 初始化数据库*/public static void setInitRealm(int version) {RealmDBConfig.version= version;if (Realm.getDefaultConfiguration() == null) {Realm.init(MyApplication.getAppContext());}configRealm(version); }private static void configRealm(int version) {ECKey ecKey = ECKey.fromPrivate(OcMath.toBigInt("0abc4301"));byte[] sha256 = Sha256Hash.hash(ecKey.getPubKey());realmConfig = new RealmConfiguration.Builder()//设置数据库升级.migration(migrationDb)//设置数据库密码.encryptionKey(OcMath.toHexStringNoPrefix(sha256).getBytes())  .name("test.realm")//是否允许在UI线程中操作写入数据库.allowWritesOnUiThread(true)//设置数据库版本.schemaVersion(version).build();Realm.setDefaultConfiguration(realmConfig);}public static Realm getRealm() {return Realm.getInstance(getRealmConfiguration());}public static RealmConfiguration getRealmConfiguration() {if (realmConfig == null) {setWalletOCRealm();}return realmConfig;}/*** 数据库版本管理*/protected static RealmMigration migrationDb = new RealmMigration() {//升级数据库@Overridepublic void migrate(DynamicRealm realm, long oldVersion, long newVersion) {RealmSchema schema = realm.getSchema();//给已有表新增字段if (oldVersion == 1) {schema.get("Rm_User").addField("nickname",String.class);}//新增表if (oldVersion == 2) {schema.create("Rm_Order").addField("orderId", String.class).addField("name", String.class).addField("number", int.class);}//删除字段if (oldVersion == 3) {schema.get("Rm_Wallet").removeField("hash", String.class);}//删除表if (oldVersion == 4) {schema.remove("Rm_Contract");}oldVersion++;}};}

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()}@JvmStaticfun 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 = oldVersionval schema = realm.schemaif (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 = oldVersionval schema = realm.schemaif (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 = oldVersionval schema = realm.schemaif (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即可。
 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/238868.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[Angular] 笔记 7:模块

Angular 中的模块(modules) 是代码在逻辑上的最大划分&#xff0c;它类似于C, C# 中的名字空间&#xff1a; module 可分为如下几种不同的类型&#xff1a; 使用模块的第一个原因是要对代码进行逻辑上的划分&#xff0c;第二个非常重要的原因是为了实现懒惰加载(lazy loading)&…

面试每日三题

MySQL篇 MySQL为什么使用B树索引 B树每个节点可以包含关键字和对应的指针&#xff0c;即B树的每个节点都会存储数据&#xff0c;随机访问比较友好&#xff0c;B树的叶子节点之间是无指针相连接的 B树所有关键字都存储在叶子节点上&#xff0c;非叶子节点只存储索引列和指向子…

计算机网络 应用层上 | 域名解析系统DNS 文件传输协议FTP,NFS 万维网URL HTTP HTML

文章目录 1 域名系统DNS1.1 域名vsIP&#xff1f;1.2 域名结构1.3 域名到IP的解析过程域名服务器类型 2 文件传送协议2.1 FTP 文件传输协议2.2 NFS 协议2.3 简单文件传送协议 TFTP 3 万维网WWW3.1 统一资源定位符URL3.2 超文本传送协议HTTP3.2.1 HTTP工作流程3.2.2 HTTP报文结构…

真实进行软件测试面试中,自动化测试面试到底会问那些?

作者&#xff1a;川石信息 链接&#xff1a;https://www.zhihu.com/question/342170872/answer/813076226 来源&#xff1a;知乎 著作权归作者所有。商业转载请联系作者获得授权&#xff0c;非商业转载请注明出处。 自动化测试面试1&#xff1a; 1、使用什么测试框架做的上…

7.串口通信uart编写思路及自定义协议

前言&#xff1a; 串口是很重要的&#xff0c;有许多模块通信接口就是串口&#xff0c;例如gps模块&#xff0c;蓝牙模块&#xff0c;wifi模块还有一些精度比较高的陀螺仪模块等等&#xff0c;所以学会了串口之后&#xff0c;这些听起来很牛批的模块都能够用起来了。此外&#…

MySQL 8.0 InnoDB Tablespaces之File-per-table tablespaces(单独表空间)

文章目录 MySQL 8.0 InnoDB Tablespaces之File-per-table tablespaces&#xff08;单独表空间&#xff09;File-per-table tablespaces&#xff08;单独表空间&#xff09;相关变量&#xff1a;innodb_file_per_table使用TABLESPACE子句指定表空间变量innodb_file_per_table设置…

Git系统有哪些优势

在现在的这个软件开发领域&#xff0c;版本控制是一项非常重要的工作。Git作为比较流行的分布式版本控制系统&#xff0c;他有着独特的优势成为了很多开发者们的首选。那Git系统都有哪些优势呢&#xff0c;下面我以自己的理解简单的介绍一下。 分布式版本控制的优势 Git用的是…

标准地址门牌管理系统:提升地址管理效率与准确性的关键

在信息化社会的今天&#xff0c;地址管理的重要性日益凸显。无论是商业活动、物流配送&#xff0c;还是公共安全&#xff0c;都需要精确、高效的地址管理。然而&#xff0c;传统地址管理方式往往存在地址不规范、信息不全等问题&#xff0c;这无疑增加了管理难度和工作量。为此…

linux 中 C++的环境搭建以及测试工具的简单介绍

文章目录 makefleCMakegdb调试 与 coredumpValgrind 内存检测gtest 单元测试 makefile 介绍 安装 : sudo apt install make makefile 的规则: 举例说明 包括&#xff1a;目标文件 、 依赖文件 、 生成规则 使用 &#xff1a; make make clean CMake : CMake是一个…

046.Python包和模块_导入相关

我 的 个 人 主 页&#xff1a;&#x1f449;&#x1f449; 失心疯的个人主页 &#x1f448;&#x1f448; 入 门 教 程 推 荐 &#xff1a;&#x1f449;&#x1f449; Python零基础入门教程合集 &#x1f448;&#x1f448; 虚 拟 环 境 搭 建 &#xff1a;&#x1f449;&…

『番外篇五』SwiftUI 进阶之如何动态获取任意视图的 tag 和 id 值

概览 在某些场景下,我们需要用代码动态去探查 SwiftUI 视图的信息。比如任意视图的 id 或 tag 值: 如上图所示:我们通过动态探查技术在运行时将 SwiftUI 特定视图的 tag 和 id 值显示在了屏幕上。 这是如何做到的呢? 在本篇博文,您将学到如下内容: 概览1. “如意如意,…

手敲MyBatis(十三章)-返回Insert操作自增索引值

1.目的 这一章的目的主要是插入语句以后返回插入记录的id&#xff0c;因为插入语句可分为要返回记录id&#xff0c;不要返回记录id的以及不同数据源类型执行的时机也不同&#xff08;如&#xff1a;oracle不支持主键&#xff0c;需要先插入序列再增加&#xff0c;Mysql支持主键…

SQL指南:掌握日期函数来查询和管理数据

文章目录 1. 引言2. 建立数据库表2.1 建表语句2.2 数据插入 查询案例3.1 查询当前日期的订单3.2 查询过去一周内的订单3.3 查询明天的日期3.4 查询今年的订单3.5 查询特定月份的订单 总结 1. 引言 在数据库管理中&#xff0c;处理日期和时间是一项基本但重要的任务。本指南将通…

数智金融技术峰会|数新网络受邀分享《金融信创湖仓一体数据平台架构实践》,敬请期待

12月23日&#xff0c;数新网络参加DataFunSummit 2023&#xff1a;数智金融技术峰会。会上&#xff0c;数新CTO原攀峰将为大家带来《金融信创湖仓一体数据平台架构实践》 主题分享。 本次峰会由DataFun联合火山引擎、蓝驰等知名企业举办&#xff0c;将共同为大家带来一场数智金…

玩转Instagram Shop只需要学会这些功能

Instagram Shop作为Instagram下属的电商购物平台。用户可以通过浏览Instagram上的推荐产品和品牌&#xff0c;在无需离开应用的情况下了解并购买新的商品。对于经常使用Instagram的用户来说是个很便捷的购物渠道。面对这个新渠道&#xff0c;我们又该如何玩转它呢。这篇文章就会…

【沐风老师】3dMax篮球建模方法详解

3dMax足球、排球和篮球建模系列之&#xff1a;篮球建模。对于足球和排球建模&#xff0c;思路是从一个基础模型开始&#xff0c;利用这个基础模型与最终的足球&#xff08;或排球&#xff09;模型的某些相似之处&#xff0c;经过修改编辑&#xff0c;最终完成目标模型的建模。但…

ansible的playbook

1、playbook的组成部分 &#xff08;1&#xff09;task任务&#xff1a;在目标主机上执行的操作&#xff0c;使用模块定义这些操作&#xff0c;每个任务都是一个模块的调用 &#xff08;2&#xff09;variables变量&#xff1a;存储和传递数据&#xff08;变量可以自定义&…

Java可变参数(学习推荐版,通俗易懂)

定义 可变参数本质还是一个数组 示例代码 注意事项 1.形参列表中&#xff0c;可变参数只能有一个 2.可变参数必须放在形参列表的最后面 注意是最后面。 name也可以为int类型

【C#】TimeSpan

文章目录 概述属性时间计算拓展来源 概述 TimeSpan结构&#xff1a;表示一个时间间隔。 它含有以下四个构造函数&#xff1a; TimeSpan(Int64)将 TimeSpan结构的新实例初始化为指定的刻度数。&#xff08;DateTime.Tick:是计算机的一个计时周期&#xff0c;单位是一百纳秒&…

3. 行为模式 - 迭代器模式

亦称&#xff1a; Iterator 意图 迭代器模式是一种行为设计模式&#xff0c; 让你能在不暴露集合底层表现形式 &#xff08;列表、 栈和树等&#xff09; 的情况下遍历集合中所有的元素。 问题 集合是编程中最常使用的数据类型之一。 尽管如此&#xff0c; 集合只是一组对象的…