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 = ; i < ; i++) { UserModel *user = [NSEntityDescription insertNewObjectForEntityForName:@"UserModel" inManagedObjectContext:_context]; //2.设置对象的属性
user.username = [NSString stringWithFormat:@"xiaomin%i", i];
user.password = @"";
user.age = @( + 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 = @"";
} //保存对象
[_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,其数组的先后顺序就是排序规则的指定,当第一个相同,再进行第二个排序,举一个生活中的例子,
当两个人的成绩相同的时候,不可能给两个人相同的名次,只能再通过另外的一种排序约束来进行排序,这是学号就能体现出来了,因为每个人的学号都是唯一的
不可改变的,而且在建立的时候就是按照数字的顺序进行排序的,这样两个排序条件就能决定具体哪个同学的排名可以排在前面。