1、概述

Core Data框架提供了对象-关系映射(ORM)的功能,即能够将OC对象转化成数据,保存在SQLite3数据库文件中,也能够将保存在数据库中的数据还原成OC对象。在此数据操作期间,不需要编写任何SQL语句。使用此功能,要添加CoreData.framework和导入主头文件<CoreData/CoreData.h>。

2、模型文件

在Core Data,需要进行映射的对象称为实体(entity),而且需要使用Core Data的模型文件来描述应用的所有实体和实体属性。

这里以Person和Card(身份证)2个实体为例子,先看看实体属性和之间的关联关系。

Person中有个Card属性,Card中有个Person属性。

属于一对一双向关联。

使用方法:

创建文件:

添加实体:

添加Person实体的基本属性:

添加Card实体的基本属性:

在Person中添加card属性:

在Card中添加person属性:

3NSManagedObject

通过Core Data从数据库取出的对象,默认情况下都是NSManagedObject对象,NSManagedObject的工作模式有点类似于NSDictionary对象,通过键-值对来存取所有的实体属性。

(1)setValue:forKey: 存储属性值(属性名为key)

(2)valueForKey: 获取属性值(属性名为key)

4Core Data主要对象

5、搭建Core Data上下文环境

从应用程序包中加载模型文件:

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];

【备注】持久化存储库的类型:

  • NSSQLiteStoreType  SQLite数据库
  • NSBinaryStoreType  二进制平面文件
  • NSInMemoryStoreType 内存库,无法永久保存数据

虽然这3种类型的性能从速度上来说都差不多,但从数据模型中保留下来的信息却不一样。

在几乎所有的情景中,都应该采用默认设置,使用SQLite作为持久化存储库

6、添加数据

传入上下文,创建一个Person实体对象:

NSManagedObject *person =

[NSEntityDescription insertNewObjectForEntityForName:@"Person" inManagedObjectContext:context];

设置简单属性:

[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],就能将更改的数据同步到数据库

7、查询数据

初始化一个查询请求:

NSFetchRequest *request = [[[NSFetchRequest alloc] init] autorelease];

设置要查询的实体:

NSEntityDescription *desc = [NSEntityDescription entityForName:@"Person" inManagedObjectContext:context];

设置排序(按照age降序):

NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:@"age" ascending:NO];

request.sortDescriptors = [NSArray arrayWithObject:sort];

设置条件过滤(name like '%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"]

}

8、删除数据

传入需要删除的实体对象:

[context deleteObject:managedObject];

将结果同步到数据库:

NSError *error = nil;

[context save:&error];

if (error) {

[NSException raise:@"删除错误" format:@"%@", [error localizedDescription]];

}

9、打开Core DataSQL日志输出开关

1.打开Product,点击Edit Scheme...

2.点击Arguments,在Arguments Passed On Launch中添加2项

  • -com.apple.CoreData.SQLDebug
  • 1
  • 10Core Data的延迟加载

    Core Data不会根据实体中的关联关系立即获取相应的关联对象,比如通过Core Data取出Person实体时,并不会立即查询相关联的Card实体;当应用真的需要使用Card时,才会查询数据库,加载Card实体的信息。

    11、    创建NSManagedObject的子类

    默认情况下,利用Core Data取出的实体都是NSManagedObject类型的,能够利用键-值对来存取数据,但是一般情况下,实体在存取数据的基础上,有时还需要添加一些业务方法来完成一些其他任务,那么就必须创建NSManagedObject的子类。

  • 选择模型文件:

  • 选择需要创建子类的实体:

  • Person.h – Person.m

    Card.h – Card.m

  • 那么生成一个Person实体对象就应该这样写:

    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;

iOS开发之数据存储之Core Data的更多相关文章

  1. 李洪强iOS开发之数据存储

    李洪强iOS开发之数据存储 iOS应用数据存储的常用方式 1.lXML属性列表(plist)归档 2.lPreference(偏好设置) 3.lNSKeyedArchiver归档(NSCoding) ...

  2. IOS 数据存储之 Core Data详解

    Core Date是ios3.0后引入的数据持久化解决方案,它是是苹果官方推荐使用的,不需要借助第三方框架.Core Date实际上是对SQLite的封装,提供了更高级的持久化方式.在对数据库操作时, ...

  3. IOS 数据存储之 Core Data具体解释

    Core Date是ios3.0后引入的数据持久化解决方式,它是是苹果官方推荐使用的.不须要借助第三方框架.Core Date实际上是对SQLite的封装.提供了更高级的持久化方式.在对数据库操作时, ...

  4. ios开发之数据存储

    iOS应用数据存储的常用方式 XML属性列表(plist)归档 Preference(偏好设置) NSKeyedArchiver归档(NSCoding) SQLite3 Core Data 应用沙盒 ...

  5. iOS开发之数据存储之SQLite3(包括FMDB)

    1.概述 SQLite3是一款开源的嵌入式关系型数据库,可移植性好.易使用.内存开销小. SQLite3是无类型的,意味着你可以保存任何类型的数据到任意表的任意字段中.比如下列的创表语句是合法的: c ...

  6. iOS开发之数据存储之XML属性列表(plist)归档

    1.概述 “归档”意思是持久化存储数据.plist文件是一种XML格式的文件,拓展名为plist.如果对象是NSString.NSDictionary.NSArray.NSData.NSNumber等 ...

  7. iOS开发之数据存储之NSData

    1.概述 使用archiveRootObject:toFile:方法可以将一个对象直接写入到一个文件中,但有时候可能想将多个对象写入到同一个文件中,那么就要使用NSData来进行归档对象. NSDat ...

  8. iOS开发_数据存储方式

    对于数据持久化的问题,博主并不准备在博文内放很多的代码进行更深一步解释,只是简单的介绍一下四种数据持久化,如有时间,会另外针对各个数据持久化的方法进行更进一步的阐述. 直接进入主题: 〈1.NSUse ...

  9. iOS开发之数据存储之Preference(偏好设置)

    1.概述 很多iOS应用都支持偏好设置,比如保存用户名.密码.字体大小等设置,iOS提供了一套标准的解决方案来为应用加入偏好设置功能. 每个应用都有个NSUserDefaults实例,通过它来存取偏好 ...

随机推荐

  1. LinkedList底层实现,及其数据结构实现。

    LinkedList底层的实现基于双向表 prev data next next指向下一个node的地址.prev指向上一个node. 这里的代码是LinkedList类的源码. private st ...

  2. JS - What does `void 0` mean?

    语法 void expression // or void(expression) MDN的描述 能向期望一个表达式的值是 undefined 的地方插入会产生副作用的表达式.也就是为了避免产生某种副 ...

  3. Python学习--19 网络编程

    TCP编程 Client 创建一个基于TCP连接的Socket: # coding: utf-8 import socket # 创建一个TCP连接: s = socket.socket(socket ...

  4. 完成OSS.Http底层HttpClient重构封装 支持标准库

    OSS.Http项目对于.Net Standard标准库的支持已经迁移完毕,OSS开源系列两个最底层的类库已经具备跨运行时支持的能力.由于OSS.Http类库是几年前我参照RestSharp的思路,完 ...

  5. DBMS 的个人理解

    再看自己数据库时,感觉还是有点迷茫:数据库到底是怎们工作的呢?虽然之前把代码都弄了一遍,可是效果还是不太理想. 数据库到底是怎么连到用户的程序上的呢?它的内部到底是怎么运行的呢?我研究了一段时间,获得 ...

  6. 读书笔记 effective c++ Item 12 拷贝对象的所有部分

    1.默认构造函数介绍 在设计良好的面向对象系统中,会将对象的内部进行封装,只有两个函数可以拷贝对象:这两个函数分别叫做拷贝构造函数和拷贝赋值运算符.我们把这两个函数统一叫做拷贝函数.从Item5中,我 ...

  7. 2017年2月22日-----------乱码新手自学.net 之Entity Framework 增删改

    由于我是自学的,没有人教,在网上查资料也查不到个所以然.问大神们也是爱理不理的. 所以这篇随笔纯粹源自于我自己的认识.是否真正正确我也没有把握. 如果有什么错误,请大神们给予指正 ========== ...

  8. gridView 编辑单元格获取单元格焦点位置(位于单元格的焦点位置)

    1.主要代码: private void Form1_Load(object sender, EventArgs e) { DataTable dt = new DataTable(); dt.Col ...

  9. Android系统--Binder系统具体框架分析(二)Binder驱动情景分析

    Android系统--Binder系统具体框架分析(二)Binder驱动情景分析 1. Binder驱动情景分析 1.1 进程间通信三要素 源 目的:handle表示"服务",即向 ...

  10. 读书笔记 effective c++ Item 26 尽量推迟变量的定义

    1. 定义变量会引发构造和析构开销 每当你定义一种类型的变量时:当控制流到达变量的定义点时,你引入了调用构造函数的开销,当离开变量的作用域之后,你引入了调用析构函数的开销.对未使用到的变量同样会产生开 ...