CoreData简介
1.Core Data 是数据持久化存储的最佳方式
2.数据最终的存储类型可以是:SQLite数据库,XML,二进制,内存里,或自定义数据类型
3.好处:能够合理管理内存,避免使用sql的麻烦,高效
4.构成:
(1)NSManagedObjectContext(被管理的数据上下文)
操作实际内容(操作持久层)
作用:插入数据,查询数据,删除数据
(2)NSManagedObjectModel(被管理的数据模型)
数据库所有表格或数据结构,包含各实体的定义信息
作用:添加实体的属性,建立属性之间的关系
操作方法:视图编辑器,或代码
(3)NSPersistentStoreCoordinator(持久化存储助理)
相当于数据库的连接器
作用:设置数据存储的名字,位置,存储方式,和存储时机
(4)NSManagedObject(被管理的数据记录)
相当于数据库中的表格记录
(5)NSFetchRequest(获取数据的请求)
相当于查询语句
(6)NSEntityDescription(实体结构)
相当于表格结构
(7)后缀为.xcdatamodeld的包
里面是.xcdatamodel文件,用数据模型编辑器编辑
编译后为.momd或.mom文件
CoreData简单使用
1.构建流程
新建工程,选择空白应用程序,next
勾选 Use Core Dataa选项
此时生成的工程文件AppDelegate中,会自动生成被管理的数据上下文等相关代码
2.创建数据模型
选中.xcodedatamodel对象
在右边的数据模型编辑器的底部工具栏点击Add Entity添加实体
然后给实体添加属性
点击工具栏Editor--->Creat NsManagerObject SubClass选项,实体创建完毕
3.代码数据操作
新建一个类,导入创建的实体类
在.h里面需要引入AppDelegate.h;创建一个属性,指定我们的对象;创建一个上下文对象,用于处理所有与存储相关的请求;
#import <UIKit/UIKit.h>
#import "AppDelegate.h"
@interface CoreDataTableViewController : UITableViewController
//创建一个属性,指定我们的对象
@property(nonatomic,strong)AppDelegate *myAppDelegate;
//创建一个上下文对象,用于处理所有与存储相关的请求
@property(nonatomic,strong)NSManagedObjectContext *myContext;
//创建一个数组,用于存储数组的数据源
@property(nonatomic,strong)NSMutableArray *allData;
@end
在viewDidLoad里,进行数据初始化
//进行数据初始化
AppDelegate *dele = [UIApplication sharedApplication].delegate;
self.myContext = dele.managedObjectContext;
self.allData = [NSMutableArray array];
通过CoreData读取本地所有的数据
-(void)getAllDataFromCoreData{
NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Student" inManagedObjectContext:self.myContext];
[fetchRequest setEntity:entity];
// Specify how the fetched objects should be sorted
//排序条件
NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"age"
ascending:YES];
[fetchRequest setSortDescriptors:[NSArray arrayWithObjects:sortDescriptor, nil]];
NSError *error = nil;
NSArray *fetchedObjects = [self.myContext executeFetchRequest:fetchRequest error:&error];
if (fetchedObjects == nil) {
NSLog(@"两手空空");
}
//将查询到的数据添加到数据源
[self.allData addObjectsFromArray:fetchedObjects];
//重新加载tableView
[self.tableView reloadData];
}
添加数据操作
- (IBAction)addAction:(UIBarButtonItem *)sender {
//1.创建student对象
//创建一个实体描述对象
NSEntityDescription *stuDIs = [NSEntityDescription entityForName:@"Student" inManagedObjectContext:self.myContext];
Student *stu = [[Student alloc] initWithEntity:stuDIs insertIntoManagedObjectContext:self.myContext];
stu.name = @"张三";
stu.age = arc4random() % 73 + 1;
//1.修改数据源
[self.allData addObject:stu];
//2.修改界面
NSIndexPath *path = [NSIndexPath indexPathForRow:self.allData.count - 1 inSection:0];
[self.tableView insertRowsAtIndexPaths:@[path] withRowAnimation:UITableViewRowAnimationLeft];
//将数据保存到文件中进行持久化
NSError *error = nil;
[self.myContext save:&error];
if (nil != error) {
NSLog(@"持久化存在问题");
}
[((AppDelegate *)[UIApplication sharedApplication].delegate) saveContext];
}
当我们提交请求的时候,就会调用下面的方法,更新UI
-(void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath{
if (editingStyle == UITableViewCellEditingStyleDelete) {
//1 获取当前cell代表的数据
Student *stu = self.allData[indexPath.row];
//2. 更新数据源
[self.allData removeObject:stu];
//3.更新UI
[self.tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationLeft];
//4. 将临时数据库里进行删除并进行本地持久化
[self.myContext deleteObject:stu];
[self.myContext save:nil];
}
}
点击cell的响应时间
-(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{
//1. 更新第一步:先查询
NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Student" inManagedObjectContext:self.myContext];
[fetchRequest setEntity:entity];
// Specify how the fetched objects should be sorted
NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"age"
ascending:YES];
[fetchRequest setSortDescriptors:[NSArray arrayWithObjects:sortDescriptor, nil]];
NSError *error = nil;
NSArray *fetchedObjects = [self.myContext executeFetchRequest:fetchRequest error:&error];
//修改对应的数据
Student *stu = self.allData[indexPath.row];
stu.name = @"山河故人";
//更新数据源
[self.allData removeAllObjects];
[self.allData addObjectsFromArray:fetchedObjects];
//更新界面
[self.tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationLeft];
//将修改本地持久化
[self.myContext save:nil];
}
App升级之后数据库字段或者表有更改会导致crash,CoreData的版本管理和数据迁移变得非常有用,手动写sql语句操作还是麻烦一些。
CoreData不光能操纵SQLite,CoreData和iCloud的结合也很好,如果有这方面需求的话优先考虑CoreData。
CoreData并不是直接操纵数据库,比如:使用CoreData时不能设置数据库的主键,目前仍需要手动操作。