CoreData
简介
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语句。
模型文件
Person实体中有:name(姓名)、age(年龄)、card(身份证)三个属性
Card实体中有:no(号码)、person(人)两个属性
接下来看看创建模型文件的过程:
1.选择模板
2.添加实体
3.添加Person的2个基本属性
4.添加Card的1个基本属性
5.建立Card和Person的关联关系
右图中的表示Card中有个Person类型的person属性,目的就是建立Card跟Person之间的一对一关联关系(建议补上这一项),在Person中加上Inverse属性后,你会发现Card中Inverse属性也自动补上了
了解NSManagedObject
2.NSManagedObject的工作模式有点类似于NSDictionary对象,通过键-值对来存取所有的实体属性
1> setValue:forKey:存储属性值(属性名为key)
CoreData中的核心对象
注:黑色表示类名,红色表示类里面的一个属性
开发步骤总结:
1.初始化NSManagedObjectModel对象,加载模型文件,读取app中的所有实体信息
2.初始化NSPersistentStoreCoordinator对象,添加持久化库(这里采取SQLite数据库)
3.初始化NSManagedObjectContext对象,拿到这个上下文对象操作实体,进行CRUD操作
代码实现
先添加CoreData.framework和导入主头文件<CoreData/CoreData.h>1.搭建上下文环境
- // 从应用程序包中加载模型文件
- 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];
2.添加数据到数据库
- // 传入上下文,创建一个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],就能将更改的数据同步到数据库
3.从数据库中查询数据
- // 初始化一个查询请求
- 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) {
- NSLog(@"name=%@", [obj valueForKey:@"name"]
- }
4.删除数据库中的数据
- // 传入需要删除的实体对象
- [context deleteObject:managedObject];
- // 将结果同步到数据库
- NSError *error = nil;
- [context save:&error];
- if (error) {
- [NSException raise:@"删除错误" format:@"%@", [error localizedDescription]];
- }
打开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>
- @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 从iOS 3.0 开始 (orm的原理)
用coreData 来操作sqlite 对象 -> 调用插入方法
CoreData 是Mac OS X中Cocoa API的一部分, 首次出现在Mac OS 10.4 和iOS 3.0 中. 它提供了 对象- 关系映射 (ORM) 功能. 可以将OC对象转化为数据, 保存在SQLite数据库文件中, 也能够将数据库的文件还原成OC对象. 在此期间, 我们不需要编写任何SQL语句.
可以简单的理解为Cocoa对SQLite的一层封装.
CocoaData 好处:
一: 极大的减少Model层的代码量.
二: 优化了使用SQLite时候的性能.
三: 提供了可视化设计.
NO.1 managedObjectContext 上下文对象 负责将我们要保存的对象存入到数据库中, 或者从数据库中取出数据
NO.2 managedObjectModel 管理对象模型 负责把我们在 CoreDataDemo.xcdatamodeld 文件中添加的entity(实体)转化为sqlite数据库中的表.
NO.3 persistentStoreCoordinator 持久化存储协调器 负责把要保存的数据(sqlite文件) 存储到沙盒中某个位置.
@synthesize property 让系统帮我们生成set get 方法
@dynamic name; // 不让系统生成get和set方法 需要我们自己去实现
插入数据:
// 插入操作
- (void)insertData
{
// 获取到上下文对象
AppDelegate *delegate = [UIApplication sharedApplication].delegate;
NSManagedObjectContext *context = delegate.managedObjectContext;
// 创建一个要插入的对象
NSManagedObject *people = [NSEntityDescription insertNewObjectForEntityForName:@"People" inManagedObjectContext:context];
[people setValue:@"王红五" forKey:@"name"];
[people setValue:@(28) forKey:@"age"];
NSError *error = nil;
// 执行保存的方法, 就会把刚才创建的对象保存到数据库中
BOOL isSave = [context save:&error];
if (isSave) {
NSLog(@"insert success");
}else
{
NSLog(@"insert failed %@",error);
}
// 方法二
People *people1 = [NSEntityDescription insertNewObjectForEntityForName:@"People" inManagedObjectContext:context];
people1.name = @"马六六";
people1.age = @21;
NSError *error1 = nil;
[context save:&error1];
}
查询操作:
- (void)queryData
{
// 获取到上下文对象
AppDelegate *delegate = [UIApplication sharedApplication].delegate;
NSManagedObjectContext *context = delegate.managedObjectContext;
// 创建一个查询的请求
NSFetchRequest *request = [[NSFetchRequest alloc]init];
// 设置要查询哪一个实体(表)
request.entity = [NSEntityDescription entityForName:@"People" inManagedObjectContext:context];
// 排序对象
NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:@"age" ascending:YES];
request.sortDescriptors = @[sort];
// 查询条件
request.predicate = [NSPredicate predicateWithFormat:@"name like %@",@"马*"];
// 调用查询的方法
NSError *error = nil;
NSArray *array = [context executeFetchRequest:request error:&error];
for (People *p in array) {
NSLog(@"name %@ age %@",p.name,p.age);
// [self updateData:p];
[self deleteData:p];
}
}
修改(更新操作):
- (void)updateData: (People *)people
{
// 修改具体的值
people.age = @40;
// 获取到上下文对象
AppDelegate *delegate = [UIApplication sharedApplication].delegate;
NSManagedObjectContext *context = delegate.managedObjectContext;
NSError *error = nil;
BOOL isOk = [context save:&error];
NSLog(@"%@",isOk == YES?@"保存成功":@"保存失败");
}
删除操作:
- (void)deleteData: (People *)people
{
// 获取到上下文对象
AppDelegate *delegate = [UIApplication sharedApplication].delegate;
NSManagedObjectContext *context = delegate.managedObjectContext;
[context deleteObject:people];
BOOL isOK = [context save:nil];
NSLog(@"%@",isOK == YES?@"删除成功":@"删除失败");
}
说到这里,整个Core Data框架的入门就结束了,其实Core Data还远不止这些功能,它还支持自动撤销机制,一对多关联等,这里就不一一介绍了