CoreData
2016-02-25 16:07
246 查看
简介
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的两个基本属性
4.添加Card的1个基本属性
5.建立Card和Person的关联关系
右图中的
表示Card中有个Person类型的person属性,目的就是建立Card跟Person之间的一对一关联关系(建议补上这一项),在Person中加上Inverse属性后,你会发现Card中Inverse属性也自动补上了
了解NSManagedObject
1.通过Core Data从数据库中取出的对象,默认情况都是NSManagedObject对象
2.NSManagedObject的工作模式有点类似于NSDictionary对象,通过键-值对来存取所有的实体属性
1> setValue:forKey: 存储属性名 (属性名为key)2> valueForKey: 获取属性值(属性名为key)
CoreData中的核心对象
注: 黑色表示类名,红色表示类里面的一个属性
开发步骤总结:
1.初始化NSManagedObjectModel对象,加载模型文件,读取app中的所有实体
2.初始化NSPersistentStoreCoordinator对象,添加持久化库(这里采用SQLite)
3.初始化NSManagedObjectContext对象,拿到这个上下文对象操作实体,进行CRUD
(CRUD是指在做计算处理时的增加(Create)、读取(Retrieve)(重新得到数据)、更新(Update)和删除(Delete)几个单词的首字母简写)
代码实现
先添加CoreData.framework和导入主头文件搭建上下文环境
// 从应用程序包中加载模型文件 NSManagedObjectModel *model = [NSManagedObjectModel mergedModelFromBundles:nil]; // 传入模型对象,初始化NSPersistentStoreCoordinator NSPersistentStoreCoordinator *psc = [[[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:model] autorelease]; // 构建SQLite数据库文件的路径 NSString *docs = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject]; NSURL *url = [NSURL fileURLWithPath:[docs stringByAppendingPathComponent:@"person.data"]]; // 添加持久化存储库,这里使用SQLite作为存储库 NSError *error = nil; NSPersistentStore *store = [psc addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:url options:nil error:&error]; if (store == nil) { // 直接抛异常 [NSException raise:@"添加数据库错误" format:@"%@", [error localizedDescription]]; } // 初始化上下文,设置persistentStoreCoordinator属性 NSManagedObjectContext *context = [[NSManagedObjectContext alloc] init]; context.persistentStoreCoordinator = psc; // 用完之后,记得要[context release];
添加数据到数据库
// 传入上下文,创建一个Person实体对象 NSManagedObject *person = [NSEntityDescription insertNewObjectForEntityForName:@"Person" inManagedObjectContext:context]; // 设置Person的简单属性 [person setValue:@"MJ" forKey:@"name"]; [person setValue:[NSNumber numberWithInt:27] forKey:@"age"]; // 传入上下文,创建一个Card实体对象 NSManagedObject *card = [NSEntityDescription insertNewObjectForEntityForName:@"Card" inManagedObjectContext:context]; [card setValue:@"4414241933432" forKey:@"no"]; // 设置Person和Card之间的关联关系 [person setValue:card forKey:@"card"]; // 利用上下文对象,将数据同步到持久化存储库 NSError *error = nil; BOOL success = [context save:&error]; if (!success) { [NSException raise:@"访问数据库错误" format:@"%@", [error localizedDescription]]; } // 如果是想做更新操作:只要在更改了实体对象的属性后调用[context save:&error],就能将更改的数据同步到数据库
从数据库中查询数据
// 初始化一个查询请求 NSFetchRequest *request = [[[NSFetchRequest alloc] init] autorelease]; // 设置要查询的实体 request.entity = [NSEntityDescription entityForName:@"Person" inManagedObjectContext:context]; // 设置排序(按照age降序) NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:@"age" ascending:NO]; request.sortDescriptors = [NSArray arrayWithObject:sort]; // 设置条件过滤(搜索name中包含字符串"Itcast-1"的记录,注意:设置条件过滤时,数据库SQL语句中的%要用*来代替,所以%Itcast-1%应该写成*Itcast-1*) NSPredicate *predicate = [NSPredicate predicateWithFormat:@"name like %@", @"*Itcast-1*"]; request.predicate = predicate; // 执行请求 NSError *error = nil; NSArray *objs = [context executeFetchRequest:request error:&error]; if (error) { [NSException raise:@"查询错误" format:@"%@", [error localizedDescription]]; } // 遍历数据 for (NSManagedObject *obj in objs) { d2a0 NSLog(@"name=%@", [obj valueForKey:@"name"] }
注:Core Data不会根据实体中的关联关系立即获取相应的关联对象,比如通过Core Data取出Person实体时,并不会立即查询相关联的Card实体;当应用真的需要使用Card时,才会再次查询数据库,加载Card实体的信息。这个就是Core Data的延迟加载机制
删除数据中的数据
// 传入需要删除的实体对象[context deleteObject:managedObject];
// 将结果同步到数据库
NSError *error = nil;
[context save:&error];
if (error) {
[NSException raise:@”删除错误” format:@”%@”, [error localizedDescription]];
}
创建NSManagedObject的子类
选择模型文件
选择需要创建子类的实体
创建完毕 就会多出两个子类
person.h
#import <Foundation/Foundation.h> #import <CoreData/CoreData.h> @class Card; @interface Person : NSManagedObject @property (nonatomic, retain) NSString * name; @property (nonatomic, retain) NSNumber * age; @property (nonatomic, retain) Card *card; @end
person.m
#import "Person.h" @implementation Person @dynamic name; @dynamic age; @dynamic card; @end
card.h
#import <Foundation/Foundation.h> #import <CoreData/CoreData.h> @class Person; @interface Card : NSManagedObject @property (nonatomic, retain) NSString * no; @property (nonatomic, retain) Person *person; @end
Card.m
#import "Card.h" #import "Person.h" @implementation Card @dynamic no; @dynamic person; @end
那么往数据库中添加数据的时候就应该写了:
Person *person = [NSEntityDescription insertNewObjectForEntityForName:@"Person" inManagedObjectContext:context]; person.name = @"MJ"; person.age = [NSNumber numberWithInt:27]; Card *card = [NSEntityDescription insertNewObjectForEntityForName:@”Card" inManagedObjectContext:context]; card.no = @”4414245465656"; person.card = card; // 最后调用[context save&error];保存数据
手动CoreData例子
#import "ViewController.h" #import <CoreData/CoreData.h> #import "Person.h" @interface ViewController () /** * 被管理对象的上下文 */ @property (nonatomic, strong) NSManagedObjectContext *context; @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; //准备工作: //1.创建上下文对象 self.context = [NSManagedObjectContext new]; //2.创建持久化存储协调器对象 //momd是xcdatamodel编译之后的文件名字 NSURL *dataModalURL = [[NSBundle mainBundle] URLForResource:@"Model" withExtension:@"momd"]; NSManagedObjectModel *objectModal = [[NSManagedObjectModel alloc] initWithContentsOfURL:dataModalURL]; NSPersistentStoreCoordinator *coordinator = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:objectModal]; //3.创建持久化存储对象 //参数一:如何存储数据(SQLite/xml/binary/内存) //参数二:设定存储类型的相关配置(连接数/timeout) nil使用系统默认的配置 //参数三:真正数据库文件所在的本地URL(XX/Document/person.db) //fileURLWithPath 本地 //URLWithString 远程 NSString *documentPath = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) firstObject]; NSLog(@"%@", documentPath); NSString *database = [documentPath stringByAppendingPathComponent:@"person.db"]; NSError *error = nil; NSPersistentStore *store = [coordinator addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:[NSURL fileURLWithPath:database] options:nil error:&error]; //4.绑定上下文和存储协调器之间的关系 [self.context setPersistentStoreCoordinator:coordinator]; } //操作对象编程和实体绑定好的模型对象person.h/.m - (IBAction)insertRecord:(id)sender { //insert into person (name, age, height) values ('Bob', 19, 1.85) //1.创建一个空的模型对象 Person *person = [NSEntityDescription insertNewObjectForEntityForName:@"Person" inManagedObjectContext:self.context]; //2.设置对象属性 person.name = @"Maggie"; person.age = @18; person.height = @1.85; //3.使用上下文对象保存到数据库中 NSError *error = nil; if (![self.context save:&error]) { NSLog(@"error:%@", error.userInfo); } } //查询/更新/删除操作:创建NSFecth - (IBAction)selectRecord:(id)sender { //select * from person where name = 'Maggie'; //请求对象(相当于select *from) NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:@"Person"]; //谓词对象(相当于where name = 'Maggie') NSPredicate *predicate = [NSPredicate predicateWithFormat:@"name=%@", @"Maggie"]; //把上面两个对象关联 request.predicate = predicate; //执行请求(获取带条件的记录) NSError *error = nil; //array中的类型是Person模型类 NSArray *array = [self.context executeFetchRequest:request error:&error]; } - (IBAction)updateRecord:(id)sender { } - (IBAction)deleteRecord:(id)sender { } - (void)didReceiveMemoryWarning { [super didReceiveMemoryWarning]; // Dispose of any resources that can be recreated. } @end
XCode也提供自动创建的方式 只要勾选Use CoreData即可
相关文章推荐
- Android之获取手机上的图片和视频缩略图thumbnails
- 峰回路转,Firefox 浏览器即将重返 iOS 平台
- 峰回路转,Firefox 浏览器即将重返 iOS 平台
- 数据库链接字符串查询网站
- 不可修补的 iOS 漏洞可能导致 iPhone 4s 到 iPhone X 永久越狱
- iOS 12.4 系统遭黑客破解,漏洞危及数百万用户
- 每日安全资讯:NSO,一家专业入侵 iPhone 的神秘公司
- DB2实例管理
- DB2实例管理
- [转][源代码]Comex公布JailbreakMe 3.0源代码
- 保障MySQL数据安全的14个最佳方法
- mysql问答汇集
- 第三章 数据库备份和还原
- 创建一个空的IBM DB2 ECO数据库的方法
- Access 2000 数据库 80 万记录通用快速分页类
- 开通一个数据库失败的原因的和解决办法
- 一个简单的asp数据库操作类
- CentOS下DB2数据库安装过程详解