转载:
简介
Core Data是iOS5之后才出现的一个框架,它提供了对象-关系映射(ORM)的功能,即能够将OC对象转化成数据,保存在SQLite数据库文件中,也能够将保存在数据库中的数据还原成OC对象。在此数据操作期间,我们不需要编写任何SQL语句,这个有点类似于著名的Hibernate持久化框架,不过功能肯定是没有Hibernate强大的。简单地用下图描述下它的作用:
左边是关系模型,即数据库,数据库里面有张person表,person表里面有id、name、age三个字段,而且有2条记录;
右边是对象模型,可以看到,有2个OC对象;
利用Core Data框架,我们就可以轻松地将数据库里面的2条记录转换成2个OC对象,也可以轻松地将2个OC对象保存到数据库中,变成2条表记录,而且不用写一条SQL语句。
模型文件
在
Core Data
,需要进行映射的对象称为
实体
(entity)
,而且需要使用
Core Data
的模型文件来描述app中的所有实体和实体属性。这里以Person(人)和Card(身份证)2个实体为例子,先看看实体属性和实体之间的关联关系:
Person实体中有:name(姓名)、age(年龄)、card(身份证)三个属性
Card实体中有:no(号码)、person(人)两个属性
接下来看看创建模型文件的过程:
1.选择模板
2.添加实体
3.添加Person的2个基本属性
4.添加Card的1个基本属性
5.建立Card和Person的关联关系
Person实体中有:name(姓名)、age(年龄)、card(身份证)三个属性
Card实体中有:no(号码)、person(人)两个属性
接下来看看创建模型文件的过程:
1.选择模板
2.添加实体
3.添加Person的2个基本属性
4.添加Card的1个基本属性
右图中的表示Card中有个Person类型的person属性,目的就是建立Card跟Person之间的一对一关联关系(建议补上这一项),在Person中加上Inverse属性后,你会发现Card中Inverse属性也自动补上了
了解NSManagedObject
1.通过
Core Data
从数据库取出的对象,默认情况下都是
NSManagedObject
对象
2.
NSManagedObject
的工作模式有点类似于
NSDictionary
对象,通过键
-
值对来存取所有的实体属性
1>setValue:forKey:存储属性值(属性名为key)
2>
valueForKey:
获取属性值
(
属性名为
key)
注:黑色表示类名,红色表示类里面的一个属性
开 发步骤总结:
1.初始化NSManagedObjectModel对象,加载模型文件,读取app中的所有实体信息
2.初始化NSPersistentStoreCoordinator对象,添加持久化库(这里采取SQLite数据库)
3.初始化NSManagedObjectContext对象,拿到这个上下文对象操作实体,进行CRUD操作
1.搭建上下文环境
2.添加数据到数据库
3.从数据库中查询数据
注:Core Data不会根据实体中的关联关系立即获取相应的关联对象,比如通过Core Data取出Person实体时,并不会立即查询相关联的Card实体;当应用真的需要使用Card时,才会再次查询数据库,加载Card实体的信息。这个就是Core Data的延迟加载机制
4.删除数据库中的数据
2.点击Arguments,在ArgumentsPassed On Launch中添加2项
1> -com.apple.CoreData.SQLDebug
2> 1
选择模型文件
选择需要创建子类的实体
创建完毕后,多 了2个子类
文件内容展示:
Person.h
Person.m
Card.h
Card.m
那么往数据库中添加数据的时候就应该写了:
说到这里,整个Core Data框架的入门就结束了,其实Core Data还远不止这些功能,它还支持自动撤销机制,一对多关联等,这里就不一一介绍了
CoreData中的核心对象
注:黑色表示类名,红色表示类里面的一个属性
开 发步骤总结:
1.初始化NSManagedObjectModel对象,加载模型文件,读取app中的所有实体信息
2.初始化NSPersistentStoreCoordinator对象,添加持久化库(这里采取SQLite数据库)
3.初始化NSManagedObjectContext对象,拿到这个上下文对象操作实体,进行CRUD操作
代码实现
先 添加CoreData.framework和导入主头文件<CoreData/CoreData.h>1.搭建上下文环境
- //从应用程序包中加载模型文件
- NSManagedObjectModel*model=[NSManagedObjectModelmergedModelFromBundles:nil];
- //传入模型对象,初始化NSPersistentStoreCoordinator
- NSPersistentStoreCoordinator*psc=[[[NSPersistentStoreCoordinatoralloc]initWithManagedObjectModel:model]autorelease];
- //构建SQLite数据库文件的路径
- NSString*docs=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,NSUserDomainMask,YES)lastObject];
- NSURL*url=[NSURLfileURLWithPath:[docsstringByAppendingPathComponent:@"person.data"]];
- //添加持久化存储库,这里使用SQLite作为存储库
- NSError*error=nil;
- NSPersistentStore*store=[pscaddPersistentStoreWithType:NSSQLiteStoreTypeconfiguration:nilURL:urloptions:nilerror:&error];
- if(store==nil){//直接抛异常
- [NSExceptionraise:@"添加数据库错误"format:@"%@",[errorlocalizedDescription]];
- }
- //初始化上下文,设置persistentStoreCoordinator属性
- NSManagedObjectContext*context=[[NSManagedObjectContextalloc]init];
- context.persistentStoreCoordinator=psc;
- //用完之后,记得要[contextrelease];
2.添加数据到数据库
- //传入上下文,创建一个Person实体对象
- NSManagedObject*person=[NSEntityDescriptioninsertNewObjectForEntityForName:@"Person"inManagedObjectContext:context];
- //设置Person的简单属性
- [personsetValue:@"MJ"forKey:@"name"];
- [personsetValue:[NSNumbernumberWithInt:27]forKey:@"age"];
- //传入上下文,创建一个Card实体对象
- NSManagedObject*card=[NSEntityDescriptioninsertNewObjectForEntityForName:@"Card"inManagedObjectContext:context];
- [cardsetValue:@"4414241933432"forKey:@"no"];
- //设置Person和Card之间的关联关系
- [personsetValue:cardforKey:@"card"];
- //利用上下文对象,将数据同步到持久化存储库
- NSError*error=nil;
- BOOLsuccess=[contextsave:&error];
- if(!success){
- [NSExceptionraise:@"访问数据库错误"format:@"%@",[errorlocalizedDescription]];
- }
- //如果是想做更新操作:只要在更改了实体对象的属性后调用[contextsave:&error],就能将更改的数据同步到数据库
3.从数据库中查询数据
- //初始化一个查询请求
- NSFetchRequest*request=[[[NSFetchRequestalloc]init]autorelease];
- //设置要查询的实体
- request.entity=[NSEntityDescriptionentityForName:@"Person"inManagedObjectContext:context];
- //设置排序(按照age降序)
- NSSortDescriptor*sort=[NSSortDescriptorsortDescriptorWithKey:@"age"ascending:NO];
- request.sortDescriptors=[NSArrayarrayWithObject:sort];
- //设置条件过滤(搜索name中包含字符串"Itcast-1"的记录,注意:设置条件过滤时,数据库SQL语句中的%要用*来代替,所以%Itcast-1%应该写成*Itcast-1*)
- NSPredicate*predicate=[NSPredicatepredicateWithFormat:@"namelike%@",@"*Itcast-1*"];
- request.predicate=predicate;
- //执行请求
- NSError*error=nil;
- NSArray*objs=[contextexecuteFetchRequest:requesterror:&error];
- if(error){
- [NSExceptionraise:@"查询错误"format:@"%@",[errorlocalizedDescription]];
- }
- //遍历数据
- for(NSManagedObject*objinobjs){
- NSLog(@"name=%@",[objvalueForKey:@"name"]
- }
4.删除数据库中的数据
- //传入需要删除的实体对象
- [contextdeleteObject:managedObject];
- //将结果同步到数据库
- NSError*error=nil;
- [contextsave:&error];
- if(error){
- [NSExceptionraise:@"删除错误"format:@"%@",[errorlocalizedDescription]];
- }
打开CoreData的SQL语句输出开关
1.打开Product,点击EditScheme...2.点击Arguments,在ArgumentsPassed On Launch中添加2项
1> -com.apple.CoreData.SQLDebug
2> 1
创建NSManagedObject的子类
默认情况下,利用Core Data取出的实体都是NSManagedObject类型的,能够利用键-值对来存取数据。但是一般情况下,实体在存取数据的基础上,有时还需要添加一些业务方法来完成一些其他任务,那么就必须创建NSManagedObject的子类选择模型文件
选择需要创建子类的实体
创建完毕后,多 了2个子类
文件内容展示:
Person.h
- #import<Foundation/Foundation.h>
- #import<CoreData/CoreData.h>
- @classCard;
- @interfacePerson:NSManagedObject
- @property(nonatomic,retain)NSString*name;
- @property(nonatomic,retain)NSNumber*age;
- @property(nonatomic,retain)Card*card;
- @end
Person.m
- #import"Person.h"
- @implementationPerson
- @dynamicname;
- @dynamicage;
- @dynamiccard;
- @end
Card.h
- #import<Foundation/Foundation.h>
- #import<CoreData/CoreData.h>
- @classPerson;
- @interfaceCard:NSManagedObject
- @property(nonatomic,retain)NSString*no;
- @property(nonatomic,retain)Person*person;
- @end
Card.m
- #import"Card.h"
- #import"Person.h"
- @implementationCard
- @dynamicno;
- @dynamicperson;
- @end
那么往数据库中添加数据的时候就应该写了:
- Person*person=[NSEntityDescriptioninsertNewObjectForEntityForName:@"Person"inManagedObjectContext:context];
- person.name=@"MJ";
- person.age=[NSNumbernumberWithInt:27];
- Card*card=[NSEntityDescriptioninsertNewObjectForEntityForName:@”Card"inManagedObjectContext:context];
- card.no=@”4414245465656";
- person.card=card;
- //最后调用[contextsave&error];保存数据
说到这里,整个Core Data框架的入门就结束了,其实Core Data还远不止这些功能,它还支持自动撤销机制,一对多关联等,这里就不一一介绍了
CoreData是CoCoa中处理数据绑定数据的关键特性,提供完整的对象持久化存储方案。如果你使用sqlite3厌倦了敲打sql语句,CoreData正解决了你这烦恼。sqlite3是CoreData处理的数据类型之一,当你将CoreData和sqlite的结合起来使用的话,你将能开发出强大的数据库应用。
CoreData与多线程
为了在查询数据的时候不让界面停滞,使用多线程是不可避免,一般我们会用thread,串行线程或者并发线程。
coredata与多线程交互的时候,每个线程都必须拥有一个manager context对象,一般有两种方式:
1.每一个线程使用私有的manager context,共享一个 persistent store coordinator
2.每个线程使用私有的manager context和私有的persistent store coordinator
对于这两种方式,我们比较推荐使用第一钟方式,因为使用第二种方式的会消耗我们更多的内存,所以推荐使用第一种。
CoreData里面还带有一个通知
NSManagedObjectContextDi<wbr><wbr>dSaveNotification</wbr></wbr>
,主要监听NSManagedObjectContext的数据是否改变,并合并数据改变到相应context
- [[NSNotificationCenter<wbr>defaultCenter]<wbr><wbr></wbr></wbr></wbr>
- <wbr><wbr><wbr><wbr>addObserverForName:NSManagedObjectContextDi<wbr>dSaveNotification<wbr></wbr></wbr></wbr></wbr></wbr></wbr>
- <wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr>object:nil<wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>
- <wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr>queue:nil<wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>
- <wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr>usingBlock:^(NSNotification*<wbr>note)<wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>
- {<wbr></wbr>
- <wbr><wbr><wbr><wbr>NSManagedObjectContext<wbr>*moc<wbr>=<wbr>self.mainManagedObjectContext<wbr>;<wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>
- <wbr><wbr><wbr><wbr><span style="color:rgb(0,102,153)"><strong>if</strong></span><wbr>(note.object<wbr>!=<wbr>moc)<wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>
- <wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr>[moc<wbr>performBlock:^(){<wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>
- <wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr>[moc<wbr>mergeChangesFromContextD<wbr>idSaveNotification:note];<wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>
- <wbr><wbr><wbr><wbr><wbr><wbr><wbr><wbr>}];<wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr>
- <wbr><wbr><wbr><wbr>}];<wbr></wbr></wbr></wbr></wbr></wbr>
- }];