接上篇。。。

4. 可空属性&默认值&忽略属性

默认情况下, 属性值可空, 如果强制要求某个属性非空, 可以使用如下方法:

遵循协议方法

+ (NSArray *)requiredProperties {

return @[@"name"];

}

特点:如果再次赋值为nil, 则会抛出异常错误

也可以设置默认值

+ (NSDictionary *)defaultPropertyValues {

return @{@"name": @""};

}

忽略属性:不想存储的某些属性,其实现方法为

+ (NSArray *)ignoredProperties

开发经验:

可以借助忽略属性&只读属性 打造计算属性, 完成集合以及UIImage对象的存储与获取

5. 通知

Realm 实例将会在每次写入事务提交后,给其他线程上的 Realm 实例发送通知

5.1. 获取 Realm 通知

token = [realm addNotificationBlock:^(NSString *notification, RLMRealm * realm) {

// 接收到更改通知, 需要做的事情

}];

5.2. 移除通知

[token stop];

注意:必须持有返回的token

6. Realm数据库

实现方案

不同的用户, 使用不同的数据库

 + (void)setDefaultRealmForUser:(NSString *)username {

    RLMRealmConfiguration *config = [RLMRealmConfiguration defaultConfiguration];

     // 使用默认的目录,但是使用用户名来替换默认的文件名

    config.fileURL= [[[config.fileURL URLByDeletingLastPathComponent]  URLByAppendingPathComponent:username]

                      URLByAppendingPathExtension:@"realm"]]];

     // 将这个配置应用到默认的 Realm 数据库当中

     [RLMRealmConfiguration setDefaultConfiguration:config];

 }

只读方式打开数据库

RLMRealmConfiguration *config = [RLMRealmConfiguration defaultConfiguration];

 // 获取需要打包文件的 URL 路径

 config.fileURL = [[NSBundle mainBundle] URLForResource:@"MyBundledData" withExtension:@"realm"];

 // 以只读模式打开文件,因为应用数据包并不可写

 config.readOnly = YES;

 // 通过配置打开 Realm 数据库

 RLMRealm *realm = [RLMRealm realmWithConfiguration:config error:nil];

 // 从打包的 Realm 数据库中读取某些数据

 RLMResults<Dog *> *dogs = [Dog objectsInRealm:realm where:@"age > 5"];

数据库文件删除

注意: 需要删除数据库文件以及辅助文件

代码实战:

NSFileManager *manager = [NSFileManager defaultManager];

RLMRealmConfiguration *config = [RLMRealmConfiguration defaultConfiguration];

NSArray<NSURL *> *realmFileURLs = @[

  config.fileURL,

  [config.fileURL URLByAppendingPathExtension:@"lock"],

  [config.fileURL URLByAppendingPathExtension:@"log_a"],

  [config.fileURL URLByAppendingPathExtension:@"log_b"],

  [config.fileURL URLByAppendingPathExtension:@"note"]

];

for (NSURL *URL in realmFileURLs) {

  NSError *error = nil;

  [manager removeItemAtURL:URL error:&error];

  if (error) {

    // 处理错误

  }

}

7. 数据库迁移

适用于修改了数据模型的情况,这里分 数据结构迁移 以及 数据迁移 属性重命名 多版本增量式迁移 四个模块分别进行说明。

  • 数据结构迁移
// 在 [AppDelegate didFinishLaunchingWithOptions:] 中进行配置

RLMRealmConfiguration *config = [RLMRealmConfiguration defaultConfiguration];

// 设置新的架构版本。这个版本号必须高于之前所用的版本号(如果您之前从未设置过架构版本,那么这个版本号设置为 0)

config.schemaVersion = ;

// 设置闭包,这个闭包将会在打开低于上面所设置版本号的 Realm 数据库的时候被自动调用

config.migrationBlock = ^(RLMMigration *migration, uint64_t oldSchemaVersion) {

  // 目前我们还未进行数据迁移,因此 oldSchemaVersion == 0

  if (oldSchemaVersion < ) {

// 什么都不要做!Realm 会自行检测新增和需要移除的属性,然后自动更新硬盘上的数据库架构

};

  }

// 告诉 Realm 为默认的 Realm 数据库使用这个新的配置对象

[RLMRealmConfiguration setDefaultConfiguration:config];

// 现在我们已经告诉了 Realm 如何处理架构的变化,打开文件之后将会自动执行迁移

[RLMRealm defaultRealm];
  • 数据迁移
// enumerateObjects:block: 方法遍历了存储在 Realm 文件中的每一个“Person”对象

[migration enumerateObjects:Person.className block:^(RLMObject *oldObject, RLMObject *newObject) {

// 将名字进行合并,存放在 fullName 域中

      newObject[@"fullName"] = [NSString stringWithFormat:@"%@ %@", oldObject[@"firstName"], oldObject[@"lastName"]];

    }];
  • 属性重命名

[migration renamePropertyForClass:Person.className oldName:@"yearsSinceBirth" newName:版本0

  • 多版本增量式迁移

假设有如下3个数据库版本:


版本0
       // v0

        @interface Person : RLMObject

        @property NSString *firstName;

        @property NSString *lastName;

        @property int age;

        @end

版本1

        // v1

        @interface Person : RLMObject

        @property NSString *fullName; // 新属性

        @property int age;

         @end
版本2 // v2 @interface Person : RLMObject @property NSString *fullName; @property NSString *email; // 新属性 @property int age; @end

迁移核心代码

if (oldSchemaVersion < ) { 

      newObject[@"fullName"] = [NSString stringWithFormat:@"%@ %@",

oldObject[@"firstName"],  oldObject[@"lastName"]];

}

// 只有当 Realm 数据库的架构版本为 0 的时候,才添加 “fullName” 属性

 [migration enumerateObjects:Person.className

block:^(RLMObject *oldObject, RLMObject *newObject) {

         if (oldSchemaVersion < ) {

newObject[@"email"] = @"";

  }
}];

Realm数据持久化方案的简单介绍和使用(二)的更多相关文章

  1. iOS: 数据持久化方案

    数据持久化方案(如果总结不到位,或者有误的地方,敬请斧正) 一.功能: 主要是将数据持久化到本地,减少对网络请求的次数,既节省了用户的流量,也增强了App的体验效果. 二.种类:  plist存储:使 ...

  2. iOS之数据持久化方案

    概论 所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启后可以继续访问之前保存的数据.在iOS开发中,有很多数据持久化的方案,接下来我将尝试着介绍一下5种方案: plist文件(属性列表) ...

  3. iOS中几种数据持久化方案

    概论 所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启后可以继续访问之前保存的数据.在iOS开发中,有很多数据持久化的方案,接下来我将尝试着介绍一下5种方案: plist文件(属性列表) ...

  4. iOS中几种数据持久化方案:我要永远地记住你!

    http://www.cocoachina.com/ios/20150720/12610.html 作者:@翁呀伟呀 授权本站转载 概论 所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启 ...

  5. redis的数据持久化方案

    Redis的持久化方案有两种 1.Rdb方式:快照形式,定期将内存中的数据持久化到硬盘.是Redis默认的数据持久化的形式. Rdb:缺点是:数据还没有更新到磁盘上,突然断电,造成数据的不完整性. 在 ...

  6. ios数据持久化--CoreData框架的介绍和使用

    1.Core Data 是数据持久化存储的最佳方式 2.数据最终的存储类型可以是:SQLite数据库,XML,二进制,内存里,或自定义数据类型 在Mac OS X 10.5Leopard及以后的版本中 ...

  7. iOS开发数据持久化技术02——plist介绍

    有疑问的请加qq交流群:390438081 我的QQ:604886384(注明来意) 微信:niuting823 1. 简单介绍:属性列表是一种xml格式的文件.扩展名.plist: 2. 特性:pl ...

  8. K8S学习笔记之Kubernetes数据持久化方案

    在开始介绍k8s持久化存储前,我们有必要了解一下k8s的emptydir和hostpath.configmap以及secret的机制和用途. 0x00 Emptydir EmptyDir是一个空目录, ...

  9. react-native AsyncStorage 数据持久化方案

    1,AsyncStorage介绍 AsyncStorage 是一个简单的.异步的.持久化的 Key-Value 存储系统,它对于 App 来说是全局性的.它用来代替 LocalStorage. 由于它 ...

随机推荐

  1. Jquery动态增加行和删除行

    $("#add_5").click(function(){ var str_1="<tr> <td><input type=\"t ...

  2. C++_运算符重载

    什么是运算符的重载? 运算符与类结合,产生新的含义. 为什么要引入运算符重载? 作用:为了实现类的多态性(多态是指一个函数名有多种含义) 怎么实现运算符的重载? 方式:类的成员函数 或 友元函数(类外 ...

  3. RAP在Linux 上的部署

    首先讲一下,rap 是什么东西 ,rap 是一种项目的管理工具.用来管理项目的接口的. 首先是在Linux 上装了一个tomcat8  .当然jdk 的环境是必须要有的, 在rap的官网下载了war包 ...

  4. Spark性能调优之代码方面的优化

    Spark性能调优之代码方面的优化 1.避免创建重复的RDD     对性能没有问题,但会造成代码混乱   2.尽可能复用同一个RDD,减少产生RDD的个数   3.对多次使用的RDD进行持久化(ca ...

  5. 九度OJ题目1137:浮点数加法 (JAVA)使用BigDecimal四个案例都通过了,但是超时了。

    题目描述: 求2个浮点数相加的和 题目中输入输出中出现浮点数都有如下的形式: P1P2...Pi.Q1Q2...Qj 对于整数部分,P1P2...Pi是一个非负整数 对于小数部分,Qj不等于0 输入: ...

  6. .NET客户端实现Redis中的管道(PipeLine)与事物(Transactions)

    序言 Redis中的管道(PipeLine)特性:简述一下就是,Redis如何从客户端一次发送多个命令,服务端到客户端如何一次性响应多个命令. Redis使用的是客户端-服务器模型和请求/响应协议的T ...

  7. adt的问题An internal error has occurred. After scene creation, #init() must be called

    这个问题困扰了我好久,我也尝试去百度.google无济于事啊,让我寝食难安,太难受了,我把它贴出来,希望后人不绕弯子... 解决办法: 即可,解决这一个问题,现在酣畅淋漓,挥洒自如的capy代码了

  8. 不可重入定时器Newlife.TimerX

    在.net常用的定时器类有下面三种,使用定时器时需要设定参数,如间断时间.定时器计溢出后的回调函数.延时.开始等,定时器的的主要方法有开始.终止等,不同的定时器实现上述的方法会有一些差异,本文会针对具 ...

  9. Java Reference 源码分析

    @(Java)[Reference] Java Reference 源码分析 Reference对象封装了其它对象的引用,可以和普通的对象一样操作,在一定的限制条件下,支持和垃圾收集器的交互.即可以使 ...

  10. SOA面向服务的架构理解

    Ø  单一应用架构 ·当网站流量很小时,只需一个应用,将所有功能都部署在一起,以减少部署节点和成本. Ø  垂直应用架构 当访问量逐渐增大,单一应用增加机器带来的加速度越来越小,将应用拆成互不相干的几 ...