Realm应用背景

Android自带的SQLite数据库,在多数场合能够满足我们的需求,但随着app广泛使用,SQLite也暴露了几个不足之处:
1、开发者编码比较麻烦,而且还要求开发者具备SQL语法知识;
2、SQLite默认没有加密功能,手机一旦丢失容易导致数据库被破解;
3、SQLite底层采用java代码,导致性能提升存在瓶颈;

基于以上几点,Android上的各种ORM应运而生(ORM全称Object Relational Mapping,即对象关系映射),最常见的便是greenDAO了。greenDAO是一个将对象映射到SQLite数据库中的ORM解决方案,它在github上的地址是https://github.com/greenrobot/greenDAO,下面是greenDAO相比直接使用SQLite的几个改进点:
1、简化数据库操作的编码,开发者可以不用熟悉SQL语法;
2、使用灵活,可在实体类中自定义类和枚举类型;
3、号称是基于SQLite的ORM框架中性能最好的;(博主没对比greenDAO与直接使用SQLite的性能差异,所以只能是跟其他ORM框架比较,比如ORMLite、sugarORM等等)
但是greenDAO使用的数据库引擎还是SQLite,因此某些方面并没有本质的改善,比如数据库的加密、数据库操作的性能等等。

对于Realm来说,这些改善就是可能的了,因为Realm有自己的数据库引擎,而且引擎使用C++编写,性能比java引擎的SQLite有数倍提升。Realm使用C++引擎还有一个好处,就是可以跨平台使用,不但能用于Android,也能用于IOS。Realm的第三个好处是,它具有很多移动设备专用数据库的特性,比如支持JSON、流式api、数据变更通知,以及加密支持,这些都为开发者带来了方便。

Realm环境搭建

Realm支持Eclipse的最后版本是0.87.5,更新的版本只支持AndroidStudio,不再支持Eclipse了,所以这里就以0.87.5为例进行说明。

0.87.5的Realm下载页面是https://realm.io/docs/java/0.87.5/#eclipse,github上最新版本的地址是https://github.com/realm/realm-java。把Realm加入到工程,除了引用realm-android-0.87.5.jar,还得加入armeabi目录下的so文件librealm-jni.so。现在编译通过了,可是运行时又坑爹了,居然报错“java.lang.IllegalArgumentException: Country is not part of the schema for this Realm”,原因是Realm采用了注解Annotation方式,所以得先让我们的Eclipse支持注解才行。类似的情况,也存在于ButterKnife这个注入框架。

按照Realm官网的说明步骤,竟然发现我们最新的ADT,在“Properties”——“Java Compiler”下并没有“Annotation Processing”。网上转悠了一圈,找到了如下解决步骤:
1、依次选择“Help”——“Install New Software”
2、下拉选择Juno,即“Juno - http://download.eclipse.org/releases/juno”
3、在Name列表中点开“Programming Languages”,然后勾选“Eclipse Java Development Tools”(最新版本是3.8.2)
4、点击“Next”按钮,执行安装操作
5、安装完毕重启ADT,就可以在“Java Compiler”下找到“Annotation Processing”了

装好Annotation插件,只是万里长征的第一步,接下来我们还得配置Eclipse,使之支持Annotation,具体步骤如下:
1、右击我们的工程,依次选择“Properties”——“Java Compiler”——“Annotation Processing”,勾选“Enable project specific settings”,并点击“Apply”按钮,然后工程会重新编译;
2、继续打开“Annotation Processing”——“Factory Path”,勾选“Enable project specific settings”,然后点击“Click Add JARs”按钮,选择工程libs目录下的realm-android-0.87.5.jar,点击“OK”按钮,然后工程又会重新编译;
3、为了确保注解的处理器一直工作,我们得在所有RealmObject派生类的前一行加上注解:@RealmClass

另外,正式的app都会进行代码混淆处理,为了避免混淆操作影响Realm的使用,我们要在proguard-project.txt增加如下配置:

-keep class io.realm.annotations.RealmModule
-keep @io.realm.annotations.RealmModule class *
-keep class io.realm.internal.Keep
-keep @io.realm.internal.Keep class * { *; }
-dontwarn javax.**
-dontwarn io.realm.**

Realm编码开发

数据库配置RealmConfiguration

RealmConfiguration是Realm的配置工具类,它采用了建造者模式来构建,下面是RealmConfiguration类的常用方法:
Builder(context) : 初始化RealmConfiguration的建造器。
Builder.name : 指定数据库的名称。如不指定默认名为default。
Builder.encryptionKey : 指定数据库的密钥。密钥可由SecureRandom的nextBytes方法获得,如不指定密钥则默认不加密。一旦建立加密的数据库,如果访问时密钥不正确,则Realm会扔出异常“java.lang.IllegalArgumentException: Illegal Argument: Invalid format of Realm file”。
Builder.schemaVersion : 指定数据库的版本号。如果不指定默认版本号为0,若原版本号与现版本号不一致,Realm会抛出异常“io.realm.exceptions.RealmMigrationNeededException: RealmMigration must be provided”。
Builder.migration : 指定迁移操作的迁移类,当Realm发现新旧版本号不一致时,会自动使用该迁移类完成迁移操作。
Builder.deleteRealmIfMigrationNeeded : 声明版本冲突时自动删除原数据库。
Builder.inMemory : 声明数据库只在内存中持久化。这意味着插入数据库后不能立即关闭数据库,因为一旦关闭数据库则内存中的数据马上丢失。若数据采用在文件中持久化,则无需担心关闭数据库导致数据丢失的问题。

build : 完成配置构建。
getRealmFolder : 获取数据库的持有者,返回File对象。
getRealmFileName : 获取数据库的文件名字符串。
getEncryptionKey : 获取数据库的加密密钥。
getSchemaVersion : 获取数据库的版本号。
getMigration : 获取迁移操作的迁移类。
shouldDeleteRealmIfMigrationNeeded : 判断是否声明版本冲突时自动删除原数据库。
getDurability : 返回数据持久化的方式

数据表对象RealmObject

RealmObject是数据表的实体基类,所有Realm的实体类都要从RealmObject派生而来。Realm实体类除了字段声明与set方法、get方法之外,还要加上若干必要的注解,举例如下:
@RealmClass : 加在类名前面,表示这是一个Realm实体类。
@PrimaryKey : 加在字段前面,表示该字段是主键。
@Required : 加在字段前面,表示该字段非空。
@Ignore: 加在字段前面,表示该字段不是Realm表的字段。因为有时我们需要处理一些额外的信息,但又不需要把这些信息保存到数据库。

下面是声明一个实体类的代码例子:

import io.realm.RealmObject;
import io.realm.annotations.Ignore;
import io.realm.annotations.PrimaryKey;
import io.realm.annotations.RealmClass;
import io.realm.annotations.Required;@RealmClass
public class Country extends RealmObject {@PrimaryKeyprivate String code;@Requiredprivate String name;@Requiredprivate int population;@Ignoreprivate String remark;public Country() {}public String getCode() {return code;}public void setCode(String code) {this.code = code;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getPopulation() {return population;}public void setPopulation(int population) {this.population = population;}public String getRemark() {return remark;}public void setRemark(String remark) {this.remark = remark;}}

数据库管理Realm

Realm是数据库管理工具类,可完成DDL和DML操作,下面是Realm类的常用方法:
getInstance : 获得一个数据库实例。可传入RealmConfiguration对象,若没有传入RealmConfiguration,则默认操作名为default.realm的数据库文件。
setDefaultConfiguration : 设置默认的RealmConfiguration配置。
deleteRealm : 删除指定配置的数据库。
isClosed : 判断数据库是否关闭。
close : 关闭数据库。
beginTransaction : 开始事务,需与commitTransaction配合使用。
commitTransaction : 结束事务,需与beginTransaction配合使用。
createObject : 从RealmObject类创建一条数据库记录,后面直接使用该类的设置方法即可写入字段值。
copyToRealm : 把指定RealmObject类插入数据库,如已存在主键相同的记录则扔出异常。
copyToRealmOrUpdate : 把指定RealmObject类插入数据库,如已存在主键相同的记录则更新原记录。
remove : 删除指定数据库记录。
executeTransaction : 单独对指定Realm执行事务,用于需要对事务失败进行处理的场合。
where : 查询指定表。返回RealmQuery对象。
distinct : 查询指定表指定记录的去重队列。返回RealmResults队列。

下面是Realm插入记录的代码示例:

         mRealm = Realm.getInstance(mConfig);mRealm.beginTransaction();Country country1 = mRealm.createObject(Country.class);country1.setName("北京");country1.setPopulation(5165800);country1.setCode("beijing");Country country2 = new Country();country2.setName("上海");country2.setPopulation(5999800);country2.setCode("shanghai");mRealm.copyToRealm(country2);Country country3 = new Country();country3.setName("福州");country3.setPopulation(876580);country3.setCode("fuzhou");mRealm.copyToRealmOrUpdate(country3);mRealm.commitTransaction();mRealm.close();

数据库查询RealmQuery

RealmQuery是数据库查询工具类,其对象由Realm的where方法获得,下面是RealmQuery类的常用方法:

查询条件
isNull : 指定字段为空。
isNotNull : 指定字段非空。
equalTo : 指定字段等于多少。
notEqualTo : 指定字段不等于多少。
greaterThan : 指定字段大于多少。
greaterThanOrEqualTo : 指定字段大等于多少。
lessThan : 指定字段小于多少。
lessThanOrEqualTo : 指定字段小等于多少。
between : 指定字段位于什么区间。
contains : 指定字段包含什么字符串。
beginsWith : 指定字段以什么字符串开头。
endsWith : 指定字段以什么字符串结尾。

返回结果集的运算结果
sum : 对指定字段求和。
average : 对指定字段求平均值。
min : 对指定字段求最小值。
max : 对指定字段求最大值。
count : 求结果集的记录数量。
findAll : 返回结果集所有字段,返回值为RealmResults队列
findAllSorted : 排序返回结果集所有字段,返回值为RealmResults队列

下面是Realm查询操作的代码示例:

         mRealm = Realm.getInstance(mConfig);RealmResults<Country> results = mRealm.where(Country.class).greaterThan("population", 1000000).findAll();String desc = String.format("找到%d条记录", results.size());for (int i = 0; i < results.size(); i++) {Country result = results.get(i);desc = String.format("%s\n其中城市%s的代码是%s,人口有%d", desc,result.getName(), result.getCode(),result.getPopulation());}tv_hello.setText(desc);if (mRealm.isClosed() != true) {mRealm.close();}

数据库迁移RealmMigration

app升级时可能伴随着数据库升级,对于Realm来说,数据库升级就是迁移操作,把原来的数据库迁移到新结构的数据库。编码中应对数据库迁移有三种方式:
1、构建RealmConfiguration时指定数据库版本号,如果原版本号与现版本号不一致,Realm会抛出异常RealmMigrationNeededException。代码中捕获异常RealmMigrationNeededException后,调用migrateRealm方法执行迁移操作,示例代码如下:

     RealmConfiguration config0 = new RealmConfiguration.Builder(this).name("default0").schemaVersion(3).build();try {realm = Realm.getInstance(config0);} catch (RealmMigrationNeededException e) {e.printStackTrace();// You can then manually call Realm.migrateRealm().Realm.migrateRealm(config0, new CustomMigration());realm = Realm.getInstance(config0);}

2、构建RealmConfiguration时指定数据库版本号,同时也指定迁移类,这样如果原版本号与现版本号不一致,Realm会自动使用迁移类执行迁移操作。示例代码如下:

     RealmConfiguration config1 = new RealmConfiguration.Builder(this).name("default1").schemaVersion(3).migration(new CustomMigration()).build();realm = Realm.getInstance(config1); // Automatically run migration if needed

3、构建RealmConfiguration时指定数据库版本号,同时声明版本冲突时自动删除原数据库,不过该方法一般不用,因为该方法会暴力删除所有数据。示例代码如下:

     RealmConfiguration config2 = new RealmConfiguration.Builder(this).name("default2").schemaVersion(3).deleteRealmIfMigrationNeeded().build();realm = Realm.getInstance(config2); // WARNING: This will delete all data in the Realm though.

点击下载本文用到的Realm数据库操作的工程代码

点此查看Android开发笔记的完整目录

Android开发笔记(八十五)手机数据库Realm相关推荐

  1. Android开发笔记(十五)淡入淡出动画TransitionDrawable

    说到淡入淡出动画,可能大家会想到补间动画里面的AlphaAnimation,不过这个深浅动画只能对透明度做渐变效果,也就是只能对一个图形做深浅的颜色变换.如果我们想要从A图片逐渐变为B图片,也就是要实 ...

  2. Android开发笔记(九十五)自定义Drawable

    Drawable Bitmap是Android对图像的定义描述,而Drawable则是对图像的展现描述,在View视图中显示图像都是通过Drawable来实现的.其中有关Bitmap的介绍参见< ...

  3. Android开发笔记(一百五十七)使用OpenGL实现翻书动画

    上一篇文章介绍了如何通过纹理渲染绘制地球仪,当然OpenGL的三维图形处理能力是很强大的,只要善于利用OpenGL,就能很方便地虚拟各种现实生活中的动画效果.本文再来谈谈使用OpenGL实现浏览电子书 ...

  4. Android开发笔记(一百五十八)运行时动态授权管理

    App开发过程中,涉及到硬件设备的操作,比如拍照.录音.定位等等,都要在AndroidManifest.xml中声明相关的权限.可是Android系统为了防止某些App滥用权限,从而允许用户在系统设置 ...

  5. Android开发笔记(一百五十四)OpenGL的画笔工具GL10

    上一篇文章介绍了OpenGL绘制三维图形的流程,其实没有传说中的那么玄乎,只要放平常心把它当作一个普通控件就好了,接下来继续介绍OpenGL具体的绘图操作,这项工作得靠三维图形的画笔GL10来完成了. ...

  6. Android开发笔记(一百五十九)Android7.0的分屏模式

    现在的手机屏幕越来越大,使得在屏幕上同时开多个窗口不再奢侈,因此Android从7.0开始顺势推出了分屏功能,也被称作多窗口模式.比如把竖长的手机屏幕分成上下两个窗口,一边在上面的窗口中观看电影,一边 ...

  7. Android开发笔记(一百五十二)H5通过WebView上传图片

    上一篇文章介绍了WebView与JS之间的数据交互,其实就是把字符串传来传去,这对文本格式的信息传输来说倒还凑合,倘若要传输图片信息就不管用了.所以,要想让h5网页支持从手机上传图片,还得另外想办法, ...

  8. Android开发笔记(十八)书籍翻页动画PageAnimation

    前面几节的动画都算简单,本文就介绍一个复杂点的动画--书籍翻页动画.Android有自带的翻页动画ViewPager,不过ViewPager只实现了平移效果.即便使用补间组合动画或者属性动画,也只是把 ...

  9. Android开发笔记(一百五十)自动识别验证码图片

    若问目前IT领域最炙手可热的技术方向,必属人工智能(简称AI)无疑.前有谷歌的阿法狗完胜围棋世界冠军柯洁,后有微软小冰出版了诗集<阳光失了玻璃窗>,一时间沸沸扬扬,似乎人工智能无所不能,从 ...

  10. Android开发笔记(一百五十六)通过渲染纹理展示地球仪

    上一篇文章介绍了如何使用GL10描绘三维物体的线段框架,后面给出的立方体和球体效果图,虽然看起来具备立体的轮廓,可离真实的物体还差得远.因为现实生活中的物体不仅仅有个骨架,还有花纹有光泽(比如衣服), ...

最新文章

  1. 中英字体不同导致的下划线不对齐问题
  2. antd 能自适应吗_ACC自适应巡航能当自动驾驶用吗?答:不能
  3. vs2008安装部署软件过程
  4. iphone字体_iOS 13终于能换花式字体了?!发在朋友圈里真的超好看!
  5. .NET运行时中的监测和可观测性
  6. 1636: Pascal山脉
  7. DTrace Tools
  8. 新浪微博时间格式解析java_仿新浪微博格式化时间
  9. matlab保存超分辨率图
  10. Win7中自带截图工具
  11. 华中科技大学网络教材
  12. 刘天佐加盟《经济适用男》 变身木讷IT精英_0
  13. 遗传算法之:八皇后问题
  14. 商务办公软件应用与实践【6】
  15. 企业微信如何发成绩给家长-Leo老师来教你
  16. 慈航公益仲恺义工大区和爱心企业助力探亲日慈善活动
  17. py0_二十一天计划书(前言以及本计划书)
  18. 网络安全运维人员面临的痛点分析
  19. JAVA 字符串数组按照ACCII码表排序
  20. 基于android7.0源码修改

热门文章

  1. Vue.use()与Vue.prototype
  2. 论文阅读笔记二十八:You Only Look Once: Unified,Real-Time Object Detection(YOLO v1 CVPR2015)...
  3. 第二:RobotFrameWork接口用例设计规范
  4. python中try...except的用法_python try...except语句、自定义异常、raise语句使用实例(异常处理的三种方法)...
  5. python实现播放音乐_python实现简易云音乐播放器
  6. python编程技术总结_Python面向对象编程(OOP)技术总结
  7. Oracel 连接远端服务器
  8. linux删除的文件有回收站么,Linux命令行删除文件到回收站
  9. linux里面的perl脚本怎么调用函数,如何在我的Perl脚本中包含另一个文件的函数?...
  10. Python中的zip函数