一. CoreData 是什么?
- Core Data 是 iOS SDK 里的一个很强大的框架,允许程序员以面向对象的方式储存和管理数据。使用 Core Data 框架,程序员可以很轻松有效地通过面向对象的接口管理数据
- Core Data 是一个模型层的技术。帮助建立代表程序状态的模型层,Core Data 也是一种 持久化技术,能将模型对象的状态持久化到磁盘,但它最重要的特点是: ** Core Data 不仅是一个加载、保存数据的框架,它还能和内存中的数据很好的共事**
- 在数据操作过程中, 无需编写任何SQL语句
- Core Data 使用包括实体和实体间关系,以及查找符合某些条件实体的请求等内容
- 开发者可以在纯对象层上查找与管理这些数据,而不必担心存储和查找的实现细节
- Core Data 框架最早出现在 Mac OS X 10.4 Tiger 与 iOS 3.0 系统,经过成千上万的应用程序以及数以百万用户的反复的验证,Core Data 确实已经是一套非常成熟的框架
- CoreData利用了Objective-C语言和运行时,巧妙地集成了CoreFoundation框架。是 一个易于使用的框架,不仅可以优雅地管理对象图,而且在内存管理方面表现异常优异
二. 怎么学习CoreData
- 第一眼看到 Core Data 令人生畏的复杂架构关系,很多人都会有无从下手的感觉.
- 可是, 一旦理解了架构图中各个部件的组成及相互之间的关系,就能体会到 Core Data API 的简洁和直观了
- Core Data stack(堆栈):如果能够理解 Core Data stack 中的各个成员所扮演的角色,那么再使用Core Data 就不会感觉到困难了
- 误区:Core Data 不是一个数据库,不要用数据库的眼光去看待 Core Data
- Core Data 不是应用程序的数据库,也不是将数据持久化保存到数据库的 API。Core Data 是一个用于管理对象图的框架。Core Data 可以把对象图写入磁盘从而持久化保存,但是这不是框架的主要目标.
三. Core Data stack
Core Data stack 是 Core Data 的核心,由一组 Core Data 核心对象组成
- NSManagedObjectContext 对象管理上下文: 负责管理模型的对象的集合
- NSManagedObjectModel 被管理的对象模型: 负责管理对象模型
- NSPersistentStoreCoordinator 存储调度器: 负责将数据保存到磁盘的
三者之间关系示意图:
分为两部分:
- 对象图管理
- 数据持久化
在这两部分的中间,即堆栈中间,是持久化存储协调器(Persistent Store Coordinator, PSC)。通过它将对象图管理部分和持久化部分绑在一起。当这两部分中的一部分需要和另一部分交互,将通过 PSC 来调节
常见的使用解决方案:
四. 创建CoreDataStack
根据 CoreDataStack 示意图,将关系以代码的形式组织起来 !
// 实例化数据模型
NSURL *modelURL = [[NSBundle mainBundle] URLForResource:modelNamewithExtension:@"momd"];
NSManagedObjectModel *model = [[NSManagedObjectModel alloc]initWithContentsOfURL:modelURL];
// 实例化持久化储存
NSPersistentStoreCoordinator *psc = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:model];
NSURL *dbURL = [[[NSFileManager defaultManager]URLsForDirectory:NSDocumentDirectory inDomains:NSUserDomainMask]lastObject];
dbURL = [dbURL URLByAppendingPathComponent:dbName];[psc addPersistentStoreWithType:NSSQLiteStoreType configuration:nil
URL:dbURL options:nil error:NULL];
// 实例化上下文
_managedObjectContext = [[NSManagedObjectContext alloc]initWithConcurrencyType:NSMainQueueConcurrencyType];
[_managedObjectContext setPersistentStoreCoordinator:psc];
[ 一些提示 ]
use scalar properties for primitive data type
在 iOS5 和 OS X10.7 之前,scalar 不能自动生成,程序员必须自己添加 setter 和 getter 的实现
@dynamic
在 Objective-C 中,如果将某个属性实现为@dynamic,意味着告诉编译器不会在编译时确定这个属性的行为实现,因此不需要在编译期间对这个属性的 getter、setter 做检查.
五. CoreData 常见操作
5.1 新增记录:
使用 NSEntityDescription 来创建对象,赋值后使用相应的 context 保存即可
5.2 删除记录:
使用 context 的 deleteObject:删除被管理的模型对象后保存即可
5.3 修改记录:
直接修改模型对象保存即可
5.4 查询记录
对使用 CoreData 进行存储的数据进行一定条件的查询后取出来使用
5.4.1 谓词(is)---NSPredicate
作用:判断条件表达式的求值返回真或假的过程
使用步骤:
- 定义 NSPredicate 对象并指定条件
- 调用谓词的 evaluateWithObject 方法判断指定条件是否满足示例:
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"self CONTAINS '1'"];
NSString *text = @"1234";
NSLog(@"%d", [predicate evaluateWithObject:text]);
案例:
- 创建 Person 的对象数组
- 查询判断姓名和年龄的过滤方法
- 传统方法
NSMutableArray *result = [NSMutableArray arrayWithCapacity:personList.count];
for (NSInteger i = 0; i < personList.count; i++)
{
Person *person = personList[i];
// 用户年龄小于5同时用户姓名中包含"1"字符串
if (person.age < 5 && NSNotFound != [person.name rangeOfString:@"1"].location)
{
[result addObject:person];
}
}
return result;
- 谓词方法
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"name CONTAINS '1' && %K BETWEEN {%d, %d}",@"age", 5, 15];
NSArray *result = [personList filteredArrayUsingPredicate:predicate];
谓词的条件指令
> < == >= <= !=
例如:
@"number >= 99"
范围运算符
IN BETWEEN
例如:
@"number BETWEEN {1,5}"
@"address IN {'shanghai','nanjing'}"
字符串本身
:SELF
例如:
@"SELF == 'APPLE'"
字符串相关:
BEGINSWITH ENDSWITH CONTAINS
例如:
@"name CONTAIN[cd] 'ang'" //包含某个字符串
@"name BEGINSWITH[c] 'sh'" //以某个字符串开头
@"name ENDSWITH[d] 'ang'" //以某个字符串结尾
- 通配符:
LIKE
例如:
@"name LIKE[cd] '*er*'" //* 代表通配符,Like 代表接受 [cd].
@"name LIKE[cd] '???er*'"
注: 星号"*" : 代表0或多个字符
问号"?" : 代表一个字符
- 正则表达式:MATCHES
例如:
NSString *regex = @"^A.+e$"; //以A开头, e结尾
@"name MATCHES %@",regex
注:[c]不区分大小写,[d]不区分发音符号即没有重音符号,[cd]既不区分大小写,也不区分发音符号.
- 合计操作
ANY SOME: 指定下列表达式中的任意元素
例如ANY children.age < 18
ALL: 指定下列表达式中的所有元素
例如ALL children.age < 18
NONE: 指定下列表达式中没有的元素
例如NONE children.age < 18
在逻辑上等于NOT(ANY ...)
IN:等于 SQL的IN
操作,左边的表达必须出现在与右边指定的集合中
例如:name IN { 'Ben','Melissa', 'Nick' }
提示 :
- 谓词中的匹配指令关键字通常使用大写字母
- 谓词中可以使用格式字符串
- 如果通过对象的
key path
指定匹配条件,需要使用 %K