iOSCoreData介绍
1.CoreData简介
Coredata用作数据持久化,使和大数据量的存储和查询
虽然是用户做数据的保存,但是并不是数据库,CoreData可以使用数据库、XML来存储数据
SQLite通过SQL语句操作语言,CoreData使用面向对象的方式进行操作数据。使用CoreData操作数据,无需使用SQL语句
在使用CoreData之前,需要导入CoreData框架
导入框架有两种方式,一个是在项目建立的时候使用CoreData,另外一个是在项目中导入CoreData,我分别演示一下怎么使用
另外一种就是在项目中导入
二、CoreData的优点
简单、易用、性能好。可以在Xcode上级逆行表的设计,或使用Instruments进行性能检测,可以直接长生高性能的代码
因为是苹果自己官方的,相对比较成熟,而且文档也想当丰富
三、CoreData的模块
NSManagedObjectConText:负责应用和数据库之间的交互
NSPersistentStoreCoorinator:添加持久化数据库(SQLite数据库),是物理数据存储的物理文件和程序之间的桥梁,负责管理不同对象上下文
NSManagedObjectMOdel:被管理的对象模型,对应定义的模型文件
NSEntityDerscriotion:实体描述
四、CoreData堆栈
在CoreData堆栈中,主要有两个部分,第一个部分是关于对象图管理,第二部分是关于数据的持久化
在两个部分之间,即堆栈中间,是持久化存储协调器(PSC),也别成为中间审查者,它将对象图管理部分和持久化部分捆绑在一起,当它们两者中的任何一个
部分需要和另外一个部分进行交流的时候,这边是需要持久化存储协调器来调节了
五、数据模型
// // ViewController.m // 01 CoreData的使用 // // Created by ZhuJiaCong on 16/7/23. // Copyright © 2016年 ZhuJiaCong. All rights reserved. // #import "ViewController.h" #import <CoreData/CoreData.h> #import "UserModel.h" @interface ViewController () { NSManagedObjectContext *_context; } @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; [self initCoreData]; // [self addUserData]; // [self fetchUser]; // [self updateUser]; [self deleteUser]; } //初始化 CoreData - (void)initCoreData { //1.创建数据库文件路径 NSString *databaseFilePath = [NSHomeDirectory() stringByAppendingPathComponent:@"Documents/UserDB.sqlite"]; //转化为URL NSURL *url = [NSURL fileURLWithPath:databaseFilePath]; NSLog(@"%@", databaseFilePath); //2.创建描述文件 //新建文件-iOS-CoreData-DataModel //点击描述文件中,左下角的 Add Entity按钮,来添加一个实体描述 //双击新生成的实体名,可以修改实体的名字 //点击 Attributes 下面的加号,来添加一条属性 //3.读取实体描述文件 NSURL *entityFilePath = [[NSBundle mainBundle] URLForResource:@"User" withExtension:@"momd"]; //通过实体描述文件创建数据模型 NSManagedObjectModel *model = [[NSManagedObjectModel alloc] initWithContentsOfURL:entityFilePath]; //4.创建数据持久化协调器 NSPersistentStoreCoordinator *psc = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:model]; //5. 添加数据持久化(创建数据库) /** * 添加一个数据持久化的方案 * * @param storeType 数据持久化的类型 * @param configuration 配置 * @param storeURL 数据持久化的保存磁盘地址 * @param options 选项 * @param error 错误 * * @return 数据持久化 */ NSError *error; [psc addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:url options:nil error:&error]; if (error) { NSLog(@"创建数据库失败:%@", error); return; } else { NSLog(@"数据库创建成功:%@", databaseFilePath); } //Context _context = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSMainQueueConcurrencyType]; //绑定PSC _context.persistentStoreCoordinator = psc; /* 数据库创建完毕后,系统会自动的创建ZUSERMODEL中的字段 ZUSERNAME,ZPASSWORD,ZAGE Z_PK 主键 primary key 系统自动定义的主键, Z_ENT 表索引,表示当前表在整个实体描述文件中的索引位置 Z_OPT 操作数,新建数据,操作数为1 每一次对数据的操作(修改),将操作数+1 */ } - (void)addUserData { //1.创建用户对象 // UserModel *user = [[UserModel alloc] init]; /* '-[UserModel setUsername:]: unrecognized selector sent to instance 0x7f91b0443b70' @dynamic 使对象不生成SET GET方法,所以不能直接设置属性 在创建对象的时候,需要使用CoreData所提供的方法,来创建SET和GET方法 使用 NSEntityDescription 来创建对象,会自动的根据实体描述中定义的属性,来创建SET和GET方法 */ for(int i = 0; i < 100; i++) { UserModel *user = [NSEntityDescription insertNewObjectForEntityForName:@"UserModel" inManagedObjectContext:_context]; //2.设置对象的属性 user.username = [NSString stringWithFormat:@"xiaomin%i", i]; user.password = @"123456"; user.age = @(18 + i); user.height = @(1.88); } //3.储存对象 将当前Context中,所有的对象,储存到数据库中 NSError *error; [_context save:&error]; if (error) { NSLog(@"保存失败:%@", error); } else { NSLog(@"保存成功"); } } //查询用户数据 - (NSArray *)fetchUser { //创建查询请求对象 NSFetchRequest *request =[NSFetchRequest fetchRequestWithEntityName:@"UserModel"]; //设置查询条件 NSPredicate *predicate = [NSPredicate predicateWithFormat:@"age>50"]; request.predicate = predicate; //执行查询请求 NSError *error; NSArray *result = [_context executeFetchRequest:request error:&error]; if (error) { NSLog(@"查询出错:%@", error); } else { NSLog(@"查询成功"); //处理查询结果 //遍历结果数组 for (UserModel *user in result) { NSLog(@"%@,%@,%@", user.username,user.password,user.age); } } return [result copy]; } - (void)updateUser { //查找年龄大于50的人 然后把他们的密码改为88888888 //查找对象 NSArray *userArray = [self fetchUser]; for (UserModel *user in userArray) { //每一个需要修改的对象 user.password = @"88888888"; } //保存对象 [_context save:nil]; } - (void)deleteUser { //查找需要删除的对象 NSArray *userArray = [self fetchUser]; //从context中 删除对象 for (UserModel *user in userArray) { //删除对象 [_context deleteObject:user]; } //保存context [_context save:nil]; } - (void)didReceiveMemoryWarning { [super didReceiveMemoryWarning]; // Dispose of any resources that can be recreated. } @end
都有详细的说明,以后用来复习复习
六、NSManagedObjectConText的常用方法
七、NSFetchedResultsController
在CoreData和UITabView之间,存在NSFetchedResultsController的类,可以在数据与现实之间搭建桥梁
NSFetchedResultsController是为了响应Model层的变化而设计的
在使用NSFetchedResultsController必须有一个sortDescirptor,而过滤predicate是可选的
执行performFetch方法可以获取缓冲好的数据
当CoreData发生变化的时候,NSFetchedResultsController可以立刻接受到响应,讲数据显示在TableView上面
注意:排序的时候可以传给一个数组给Predicate,其数组的先后顺序就是排序规则的指定,当第一个相同,再进行第二个排序,举一个生活中的例子,
当两个人的成绩相同的时候,不可能给两个人相同的名次,只能再通过另外的一种排序约束来进行排序,这是学号就能体现出来了,因为每个人的学号都是唯一的
不可改变的,而且在建立的时候就是按照数字的顺序进行排序的,这样两个排序条件就能决定具体哪个同学的排名可以排在前面。