为什么我不再使用Realm

也许你并没有听说过 Realm ,这是一个面向安卓(亦面向iOS)的移动端数据库技术。和SQLite不同,它允许你在持久层直接和数据对象工作。在它之上是一个函数式风格的查询api,众多的努力让它比传统的SQLite 操作更快 。基于这些原因让我决定试试Realm 。

大约一年前,当我第一次使用Realm 的时候,给我的第一印象非常不错。我需要保持一些用户数据在手机本地,但是SharedPreferences用起来有点复杂了。Realm允许我用快速干净的代码里完成这件事情。它完全不需要像SQLite那样自己手动写额外的代码。

我接下来的一个项目在缺少网络连接的时候需要一个比较复杂的离线模式。从网络抓取的数据必须保存在手机本地。我决定完全使用Realm 并观察它随项目增大是如何扩大的。

而我很快发现Realm让数据模型的工作成了一种负担。它有几个限制以至于我必须在整个代码基础上做处理。结果我尝试在Realm之上抽象出另外一层来减轻这种限制。

定义对象

为了说明这些限制,让我们从简单的Person对象开始:

public class Person extends RealmObject {  
    private String name;  
    private int age;    
    public String getName() { return name; }  
    public void setName(String name) { this.name = name; }    
    public int getAge() { return age; }  
    public void setAge(int age) { this.age = age; }
}

注意我们必须直接继承自RealmObject。这阻碍我们利用数据模型中的任意类型的继承。

并且我们还不能定义除setters 和 getters之外的实例方法。如果你想重写equals 或者toString那么你就别想了。这样导致的另外一个后果就是我们只能局限于使用标记接口模式(marker interfaces) (注解也是可以的 )。

不仅仅被限制于setters 和 getters,实际上我们还必须提供它们。因此我们的数据对象是不可变的!另外,setters 和 getters方法只是为Realm替换自己实现的代理方法。它不能操作数据,跑出异常,或者打印日志。

虽然我们可以提供一个非默认的构造函数,但是我们必须保证存在一个空的构造函数。如果你想用一个builder 或者工厂方法来作为实例化的唯一途径,那么这种限制就成了一个问题。稍后我们将看看如何用Realm创建对象。

在我们能持有的field类型方面,也有一些限制。所有的基本数据类型以及它们的封装类型都能支持,包括String, Date, 和byte[]`。但是对于其它类型,为了被持久化,必须继承自RealmObject。Lists可以用RealmList来支持。

但是也仅此而已。如果我们想使用枚举而不是int,是没有办法的(找到一个使用@IntDef的理由了)。我们还不能使用集合类型,比如Set和Map。

创建和更新对象

为了创建一个Person类的实例,我们必须做如下事情:

Realm realm = Realm.getInstance(context);
realm.beginTransaction();
Person person = realm.createObject(Person.class);
person.setName("John");
person.setAge(25);
realm.commitTransaction();

你会注意到我们必须包裹一下Person 对象,同时任何对它的修改都在一个transaction 中。如果我们能在transaction 之外做这件事情并在我们准备好的时候持久化它,就要灵活得多。而现在我们在想要创建或者更新我们对象的任何时候都要卡在写额外的Realm 代码上面。

之前我提过我们可以定义一个非默认的构造函数。比如,对于Person我们可能有一个带有name 和 age的构造函数:

public class Person extends RealmObject {  
    private String name;  
    private int age;    
    public Person(String name, int age) {    
        this.name = name;    
        this.age = age;  
    }    
    public Person() {    
        // required empty public constructor  
    }    
    // setters and getters  
    ...
}

我们不再需要直接调用setters :

Person person = new Person("John", 25);
realm.beginTransaction();
Person realmPerson = realm.copyToRealm(person);
realm.commitTransaction();

这让我们省去了一些写额外代码的时间,但是仍然受限于transaction。

Mitigating These Issues

为了避免在基础代码中处理这些限制,我为数据对象定义了两套类: POJOs (普通对象)和Realm 对象。然后我们创建了一个能在两者之间映射的abstraction 。

这是可行的,但是有两个主要的问题。第一个是当你持有许多不同类型的对象时,你需要许多代码来映射这些类。管理这些是很痛苦的而且这也很容易产生bug。 对象映射的概念以及它存在的问题都不是什么新东西了。

第二个就是我觉得这有违最初使用Realm的目的。能在持久层直接使用对象是它的主要好处。如果我们为了使用POJO而必须在Realm 之上创建抽象,那么它相比SQLite或者像 DBFlow一样的ORM的优势在哪里呢?

值得一提的是Realm 的维护者已经 意识到了这些限制 ,而且在一定程度上,许多问题都可能被解决(见这里这里))。Realm也的确具有一些其它的优势,比如性能以及在iOS和安卓之间共享数据的能力。

英文原文:Why I Don't Use Realm Anymore

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

推荐阅读更多精彩内容