Core Data是iOS5之后才出现的一个框架,本质上是对SQLite的一个封装,它提供了对象-关系映射(ORM)的功能,即能够将OC对象转化成数据,保存在SQLite数据库文件中,也能够将保存在数据库中的数据还原成OC对象,通过CoreData管理应用程序的数据模型,可以极大程度减少需要编写的代码数量!
更新于2018.3.7 iOS CoreData (二) 版本升级和数据库迁移
1、首先创建一个coreData 模型文件:系统创建或者自己创建
###2、在data Model 中创建项目中需要用到的实体(Entities),例如,创建一个Student 实体(第一字母必须是大写),以及添加一些name、age、sex 等属性,如下图
###3、生成对应实体的实体类,在此之前要注意下图两个设置部分,否则会引起崩溃现象
4、生成上下文 关联数据库
- NSManagedObjectContext 管理对象,上下文,持久性存储模型对象,处理数据与应用的交互
- NSManagedObjectModel 被管理的数据模型,数据结构
- NSPersistentStoreCoordinator 添加数据库,设置数据存储的名字,位置,存储方式
- NSManagedObject 被管理的数据记录
- NSFetchRequest 数据请求
- NSEntityDescription 表格实体结构
①、自己创建模型文件时需要以下代码来手动生成上下文,关联数据库
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37
|
//创建数据库 - (void)createSqlite{
//1、创建模型对象 //获取模型路径 NSURL *modelURL = [[NSBundle mainBundle] URLForResource:@"Model" withExtension:@"momd"]; //根据模型文件创建模型对象 NSManagedObjectModel *model = [[NSManagedObjectModel alloc] initWithContentsOfURL:modelURL];
//2、创建持久化存储助理:数据库 //利用模型对象创建助理对象 NSPersistentStoreCoordinator *store = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:model];
//数据库的名称和路径 NSString *docStr = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject]; NSString *sqlPath = [docStr stringByAppendingPathComponent:@"coreData.sqlite"]; NSLog(@"数据库 path = %@", sqlPath); NSURL *sqlUrl = [NSURL fileURLWithPath:sqlPath];
NSError *error = nil; //设置数据库相关信息 添加一个持久化存储库并设置类型和路径,NSSQLiteStoreType:SQLite作为存储库 [store addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:sqlUrl options:nil error:&error];
if (error) { NSLog(@"添加数据库失败:%@",error); } else { NSLog(@"添加数据库成功"); }
//3、创建上下文 保存信息 对数据库进行操作 NSManagedObjectContext *context = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSMainQueueConcurrencyType];
//关联持久化助理 context.persistentStoreCoordinator = store; _context = context;
}
|
②、系统创建模型文件时会自动生成关联数据库的代码,在iOS10以下和iOS10之后生成的不一样,出现了一个新类NSPersistentContainer。
NSPersistentContainer是一个容器,封装了应用程序中的CoreData Stack(核心数据栈堆),简化了创建和管理的核心堆栈的数据处理创建NSManagedObjectModel,NSPersistentStoreCoordinator,NSManagedObjectContext。详情可以看看这篇文章:http://blog.csdn.net/u013263917/article/details/53277708
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
|
AppDelegate * appDelegate = (AppDelegate *)[UIApplication sharedApplication].delegate;
NSPersistentContainer * container = appDelegate.persistentContainer;
//返回沙盒中存储数据库的文件夹URL路径,这是一个静态方法,表示数据库的文件路径是唯一的 NSURL * url = [NSPersistentContainer defaultDirectoryURL];
NSManagedObjectContext *viewContext = container.viewContext;
NSManagedObjectModel *managedObjectModel = container.managedObjectModel;
NSPersistentStoreCoordinator *persistentStoreCoordinator = container.persistentStoreCoordinator;
//使用存储调度器快速在多线程中操作数据库,效率非常高(比主线程操作块50倍!!!) - (void)performBackgroundTask:(void (^)(NSManagedObjectContext *))block;
|
###5.增删改查排
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
|
// 1.根据Entity名称和NSManagedObjectContext获取一个新的继承于NSManagedObject的子类Student Student * student = [NSEntityDescription insertNewObjectForEntityForName:@"Student" inManagedObjectContext:_context];
//2.根据表Student中的键值,给NSManagedObject对象赋值 student.name = [NSString stringWithFormat:@"Mr-%d",arc4random()%100]; student.age = arc4random()%20; student.sex = arc4random()%2 == 0 ? @"美女" : @"帅哥" ; student.height = arc4random()%180; student.number = arc4random()%100
// 3.保存插入的数据 NSError *error = nil; if ([_context save:&error]) { [self alertViewWithMessage:@"数据插入到数据库成功"]; }else{ [self alertViewWithMessage:[NSString stringWithFormat:@"数据插入到数据库失败, %@",error]]; }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
|
- (void)deleteData{
//创建删除请求 NSFetchRequest *deleRequest = [NSFetchRequest fetchRequestWithEntityName:@"Student"];
//删除条件 NSPredicate *pre = [NSPredicate predicateWithFormat:@"age < %d", 10]; deleRequest.predicate = pre;
//返回需要删除的对象数组 NSArray *deleArray = [_context executeFetchRequest:deleRequest error:nil];
//从数据库中删除 for (Student *stu in deleArray) { [_context deleteObject:stu]; }
NSError *error = nil; //保存--记住保存 if ([_context save:&error]) { [self alertViewWithMessage:@"删除 age < 10 的数据"]; }else{ NSLog(@"删除数据失败, %@", error); } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
|
//更新,修改 - (void)updateData{
//创建查询请求 NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:@"Student"]; NSPredicate *pre = [NSPredicate predicateWithFormat:@"sex = %@", @"帅哥"]; request.predicate = pre;
//发送请求 NSArray *resArray = [_context executeFetchRequest:request error:nil];
//修改 for (Student *stu in resArray) { stu.name = @"且行且珍惜_iOS"; }
//保存 NSError *error = nil; if ([_context save:&error]) { [self alertViewWithMessage:@"更新所有帅哥的的名字为“且行且珍惜_iOS”"]; }else{ NSLog(@"更新数据失败, %@", error); } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63
|
//读取查询 - (void)readData{
/* 谓词的条件指令 1.比较运算符 > 、< 、== 、>= 、<= 、!= 例:@"number >= 99"
2.范围运算符:IN 、BETWEEN 例:@"number BETWEEN {1,5}" @"address IN {'shanghai','nanjing'}"
3.字符串本身:SELF 例:@"SELF == 'APPLE'"
4.字符串相关:BEGINSWITH、ENDSWITH、CONTAINS 例: @"name CONTAIN[cd] 'ang'" //包含某个字符串 @"name BEGINSWITH[c] 'sh'" //以某个字符串开头 @"name ENDSWITH[d] 'ang'" //以某个字符串结束
5.通配符:LIKE 例:@"name LIKE[cd] '*er*'" //*代表通配符,Like也接受[cd]. @"name LIKE[cd] '???er*'"
*注*: 星号 "*" : 代表0个或多个字符 问号 "?" : 代表一个字符
6.正则表达式:MATCHES 例:NSString *regex = @"^A.+e$"; //以A开头,e结尾 @"name MATCHES %@",regex
注:[c]*不区分大小写 , [d]不区分发音符号即没有重音符号, [cd]既不区分大小写,也不区分发音符号。
7. 合计操作 ANY,SOME:指定下列表达式中的任意元素。比如,ANY children.age < 18。 ALL:指定下列表达式中的所有元素。比如,ALL children.age < 18。 NONE:指定下列表达式中没有的元素。比如,NONE children.age < 18。它在逻辑上等于NOT (ANY ...)。 IN:等于SQL的IN操作,左边的表达必须出现在右边指定的集合中。比如,name IN { 'Ben', 'Melissa', 'Nick' }。
提示: 1. 谓词中的匹配指令关键字通常使用大写字母 2. 谓词中可以使用格式字符串 3. 如果通过对象的key path指定匹配条件,需要使用%K
*/
//创建查询请求 NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:@"Student"]; //查询条件 NSPredicate *pre = [NSPredicate predicateWithFormat:@"sex = %@", @"美女"]; request.predicate = pre;
// 从第几页开始显示 // 通过这个属性实现分页 //request.fetchOffset = 0; // 每页显示多少条数据 //request.fetchLimit = 6;
//发送查询请求 NSArray *resArray = [_context executeFetchRequest:request error:nil];
[self alertViewWithMessage:@"查询所有的美女"]; }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
|
//排序 - (void)sort{ //创建排序请求 NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:@"Student"]; //实例化排序对象 NSSortDescriptor *ageSort = [NSSortDescriptor sortDescriptorWithKey:@"age"ascending:YES]; NSSortDescriptor *numberSort = [NSSortDescriptor sortDescriptorWithKey:@"number"ascending:YES]; request.sortDescriptors = @[ageSort,numberSort]; //发送请求 NSError *error = nil; NSArray *resArray = [_context executeFetchRequest:request error:&error]; if (error == nil) { [self alertViewWithMessage:@"按照age和number排序"]; }else{ NSLog(@"排序失败, %@", error); } }
|
#####CoreData调试:
打开Product,选择Edit Scheme.
选择Arguments,在下面的ArgumentsPassed On Launch中添加下面两个选项,如图:
(1)-com.apple.CoreData.SQLDebug
(2)1
参考文献:
http://www.jianshu.com/p/880dd63c5f5e
http://www.jianshu.com/p/79ba72810d29
http://www.jianshu.com/p/3e4f33b5b013
http://blog.csdn.net/u013263917/article/details/53277708
http://blog.csdn.net/u013263917/article/details/53277863
示例Github:CoreDataLearn
更新于2018.3.7 iOS CoreData (二) 版本升级和数据库迁移
近期评论