1、基础概念深刻 html
1.NSManagedObjectContext 数据库
被管理数据上下文就像便笺簿 express
当从数据持久层获取数据时,至关于把这些临时的数据拷贝写在便笺簿上,而后就能够为所欲为的修改这些值。 缓存
经过上下文,能够对数据记录NSManagedObject进行添加删除更改,记录更改后支持撤销和重作。 多线程
除非你保存这些数据变化,不然持久层的东西是不会变化。 app
一般咱们将 controller 类或其子类与 Managed Object Context NSManagedObjectContext绑定,这样就方便咱们动态地生成,获取数据对象等。 less
经常使用的方法: fetch
-save: | 将数据对象保存到数据文件 |
-objectWithID: | 查询指定 Managed Object ID 的数据对象 |
-deleteObject: | 将一个数据对象标记为删除,可是要等到 Context 提交更改时才真正删除数据对象 |
-undo | 回滚最后一步操做,这是都 undo/redo 的支持 |
-lock | 加锁,经常使用于多线程以及建立事务。同类接口还有:-unlock and -tryLock |
-rollback | 还原数据文件内容 |
-reset | 清除缓存的 Managed Objects。只应当在添加或删除 Persistent Stores 时使用 |
-undoManager | 返回当前 Context 所使用的 NSUndoManager |
-assignObject: toPersistantStore: | 因为 Context 能够管理从不一样数据文件而来的数据对象, 这个接口的做用就是指定数据对象的存储数据文件(经过指定 PersistantStore 实现) |
-executeFetchRequest: error: | 执行获取数据请求,返回全部匹配的数据对象 |
2.NSManagedObject ui
被管理的数据记录,至关于数据库中的一条记录 lua
每个NSManagedObject对象,都有一个全局 ID(类型为:NSManagedObjectID)。每一个在NSManagedObjectContext注册过
的NSManagedObject,能够经过这个全局 ID 在上下文中查询到。
每一个在持久存储层中的对象,都对应一个与上下文相关的NSManagedObject
经常使用的方法:
-entity 获取实体
-objectID 获取NSManagedObjectID
-valueForKey: 获取指定 Property 的值
-setValue: forKey: 设定指定 Property 的值
3.NSFetchRequest
获取数据的请求,经过被管理数据的上下文来执行查询,好比
NSArray *fetchedObjects = [context executeFetchRequest:fetchRequest error:&error];
查询时,必须指定查询实体或实体名称,以 NSArray 形式返回查询结果,若是咱们没有设置任何查询条件,则返回该 Entity 的全部数据对象。
咱们可使用谓词来设置查询条件,一般会将经常使用的 Fetch Requests 保存到 dictionary 以重复利用。
NSFetchRequest包括如下部分:
(1)实体(Entity)的名称
(2)NSPredicate谓词(搜索关键字或限定条件)
(3)排序方式(NSArray *)sortDescriptors
全部的被管理对象(managed object)都必须在上下文中注册,而经过NSFetchRequest得到的对象自动被注册。
若是在上下文中已经存在了要获取的对象,那么这个被管理NSManagedObject将被返回。不然上下文就会从相关的数据源中查找(也可能找不到)
例如,如下代码是查询在指定日期以后建立的ContactInfo,并将查询结果按照name排序
NSManagedObjectContext * context = [self managedObjectContext];
NSManagedObjectModel * model = [self managedObjectModel];
NSDictionary * entities = [model entitiesByName];
NSEntityDescription * entity = [entities valueForKey:@"ContactInfo"];
NSPredicate * predicate;
predicate = [NSPredicate predicateWithFormat:@"creationDate > %@", date];
NSSortDescriptor * sort = [[NSortDescriptor alloc] initWithKey:@"name"];
NSArray * sortDescriptors = [NSArray arrayWithObject: sort];
NSFetchRequest * fetch = [[NSFetchRequest alloc] init];
[fetch setEntity: entity];
[fetch setPredicate: predicate];
[fetch setSortDescriptors: sortDescriptors];
NSArray * results = [context executeFetchRequest:fetch error:nil];
[sort release];
[fetch release];
经常使用方法:
-setEntity: | 设置你要查询的数据对象的类型(Entity) |
-setPredicate: | 设置查询条件 |
-setFetchLimit: | 设置最大查询对象数目 |
-setSortDescriptors: | 设置查询结果的排序方法 |
-setAffectedStores: | 设置能够在哪些数据存储中查询 |
4.NSPersistentStoreCoordinator
持久化数据助理
Core Data定义了一个栈,持久化存储助理在中间,栈顶是被管理数据的上下文,栈底是持久化存储层,结构如图
一般从磁盘上的数据文件中读取或存储数据,这些底层的读写就由它来处理。通常咱们无需与它直接打交道,上下文已经封装了对它的调用
经常使用方法:
-addPersistentStoreForURL:configuration:URL:options:error: |
加载持久化存储数据,对应的卸载接口为 -removePersistentStore:error: |
-migratePersistentStore:toURL:options:withType:error: | 迁移数据存储,效果与 "save as"类似,可是操做成功后, 迁移前的数据存储不可再使用 |
-managedObjectIDForURIRepresentation: | 返回给定 URL所指示的数据存储的 object id,若是找不到匹配的数据存储则返回 nil |
-persistentStoreForURL: | 返回指定路径的 Persistent Store |
-URLForPersistentStore: | 返回指定 Persistent Store 的存储路径 |
5.NSManagedObjectModel
被管理的数据模型,用来描述程序的实体、其属性、关系的模型图
包括如下几个部分:
(1)实体(Entity)
对应NSEntityDescription对象
至关于数据库中的一个表
实体名称(name)
实体类名:NSManagedObject子类的名称
实体实例:NSManagedObject对象或其子类的实例
NSEntityDescription 经常使用方法:
+insertNewObjectForEntityForName:inManagedObjectContext: 工厂方法,
根据给定的 Entity 描述,生成相应的 NSManagedObject 对象,并插入 ManagedObjectContext 中。
-managedObjectClassName返回映射到 Entity 的 NSManagedObject 类名
-attributesByName以名字为 key, 返回 Entity 中对应的 Attributes
-relationshipsByName以名字为 key, 返回 Entity 中对应的 Relationships
(2)属性(Property)
对应NSPropertyDescription对象
Property 为 Entity 的特性,它至关于数据库表中的一列,或者 XML 文件中的 value-key 对中的 key。
它能够描述实体基本属性(Attribute),实体之间的关系(RelationShip),或查询属性(Fetched Property)。
<1> 实体的基本属性(Attributes)
对应NSAttributeDescription对象
存储基本数据,数据类型包括:
string,date,integer(NSString, NSDate, NSNumber)
<2> 实体间的关系(Relationships)
对应NSRelationshipDescription对象
支持对1、对多的关系
<3> 查询属性(Fetched Property)
对应NSFetchedPropertyDescription对象
根据查询谓词返回指定实体的符合条件的数据对象
表示了一种“弱”的、单项的关系(至关于数据库中的查询语句)
6.持久化存储层(Persistent Stores)
持久化存储层是和文件或外部数据库关联的,大多数访问持久化存储层的动做都由上下文来完成。
7.NSFetchedResultsController
用于在表视图table view中加载部分数据
2、用代码建立数据模型
NSManagedObjectModel *managedObjectModel() 1)咱们建立了一个全局模型 moModel;
2)并在其中建立一个名为 Run 的 Entity,这个 Entity 对应的 ManagedObject 类名为 Run(很快咱们将建立这样一个类);
3)给 Run Entity 添加了两个必须的 Property:date 和 processID,分别表示运行时间以及进程 ID;并设置默认的进程 ID 为 -1;
4)给 processID 特性设置检验条件:必须大于 0;
5)给模型设置本地化描述词典;
本地化描述提供对 Entity,Property,Error信息等的便于理解的描述,其可用的键值对以下表:
Key | Value |
"Entity/NonLocalizedEntityName" | "LocalizedEntityName" |
"Property/NonLocalizedPropertyName/Entity/EntityName" | "LocalizedPropertyName" |
"Property/NonLocalizedPropertyName" | "LocalizedPropertyName" |
"ErrorString/NonLocalizedErrorString" | "LocalizedErrorString" |
3、存储数据到xml文件
存储类型为NSXMLStoreType
NSManagedObjectContext *managedObjectContext()4、自定义NSManagedObject子类
好比,Run.h文件
#import <CoreData/NSManagedObject.h>Run.m文件
#import "Run.h"
@implementation Run
@dynamic date;
@dynamic primitiveDate;
- (void) awakeFromInsert
{
[super awakeFromInsert];
self.primitiveDate = [NSDate date];
}
#pragma mark -
#pragma mark Getter and setter
- (NSInteger)processID
{
[self willAccessValueForKey:@"processID"];
NSInteger pid = processID;
[self didAccessValueForKey:@"processID"];
return pid;
}
- (void)setProcessID:(NSInteger)newProcessID
{
[self willChangeValueForKey:@"processID"];
processID = newProcessID;
[self didChangeValueForKey:@"processID"];
}
// Implement a setNilValueForKey: method. If the key is “processID” then set processID to 0.
- (void)setNilValueForKey:(NSString *)key {
if ([key isEqualToString:@"processID"]) {
self.processID = 0;
}
else {
[super setNilValueForKey:key];
}
}
@end
1)这个类中的 date 和 primitiveDate 的访问属性为 @dynamic,这代表在运行期会动态生成对应的 setter 和 getter;
2)在这里咱们演示了如何正确地手动实现 processID 的 setter 和 getter:为了让 ManagedObjecContext 可以检测 processID的变化,以及自动支持 undo/redo,咱们须要在访问和更改数据对象时告之系统,will/didAccessValueForKey 以及 will/didChangeValueForKey 就是起这个做用的。
3)当咱们设置 nil 给数据对象 processID 时,咱们能够在 setNilValueForKey 捕获这个状况,并将 processID 置 0;
4)当数据对象被插入到 ManagedObjectContext 时,咱们在 awakeFromInsert 将时间设置为当前时间。
http://www.cnblogs.com/xiaodao/archive/2012/10/09/2716579.html