文档章节

Core Data入门

独-奏
 独-奏
发布于 2014/02/13 17:39
字数 1673
阅读 112
收藏 0

简介

  Core Data是iOS5之后才出现的一个框架,它提供了对象-关系映射(ORM)的功能,即能够将OC对象转化成数据,保存在SQLite数据库文件中,也能够将保存在数据库中的数据还原成OC对象。在此数据操作期间,我们不需要编写任何SQL语句,这个有点类似于著名的Hibernate持久化框架,不过功能肯定是没有Hibernate强大的。简单地用下图描述下它的作用:

abc

左边是关系模型,即数据库,数据库里面有张person表,person表里面有id、name、age三个字段,而且有2条记录;

右边是对象模型,可以看到,有2个OC对象;

利用Core Data框架,我们就可以轻松地将数据库里面的2条记录转换成2个OC对象,也可以轻松地将2个OC对象保存到数据库中,变成2条表记录,而且不用写一条SQL语句。


模型文件

  在Core Data,需要进行映射的对象称为实体(entity),而且需要使用Core Data的模型文件来描述app中的所有实体和实体属性。这里以Person(人)和Card(身份证)2个实体为例子,先看看实体属性和实体之间的关联关系:
abc
Person实体中有:name(姓名)、age(年龄)、card(身份证)三个属性
Card实体中有:no(号码)、person(人)两个属性

接下来看看创建模型文件的过程:
1.选择模板
abcabc

2.添加实体


3.添加Person的2个基本属性



4.添加Card的1个基本属性



5.建立Card和Person的关联关系

        

右图中的表示Card中有个Person类型的person属性,目的就是建立Card跟Person之间的一对一关联关系(建议补上这一项),在Person中加上Inverse属性后,你会发现Card中Inverse属性也自动补上了



了解NSManagedObject

1.通过Core Data从数据库取出的对象,默认情况下都是NSManagedObject对象
  

2.NSManagedObject的工作模式有点类似于NSDictionary对象,通过键-值对来存取所有的实体属性

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

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


CoreData中的核心对象



注:黑色表示类名,红色表示类里面的一个属性
发步骤总结:
1.初始化NSManagedObjectModel对象,加载模型文件,读取app中的所有实体信息
2.初始化NSPersistentStoreCoordinator对象,添加持久化库(这里采取SQLite数据库)
3.初始化NSManagedObjectContext对象,拿到这个上下文对象操作实体,进行CRUD操


代码实现

添加CoreData.framework和导入主头文件<CoreData/CoreData.h>


1.搭建上下文环境

// 从应用程序包中加载模型文件  
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];


2.添加数据到数据库

// 传入上下文,创建一个Person实体对象  
NSManagedObject *person = [NSEntityDescription insertNewObjectForEntityForName:@"Person" inManagedObjectContext:context];  
// 设置Person的简单属性  
[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],就能将更改的数据同步到数据库


3.从数据库中查询数据

// 初始化一个查询请求  
NSFetchRequest *request = [[[NSFetchRequest alloc] init] autorelease];  
// 设置要查询的实体  
request.entity = [NSEntityDescription entityForName:@"Person" inManagedObjectContext:context];  
// 设置排序(按照age降序)  
NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:@"age" ascending:NO];  
request.sortDescriptors = [NSArray arrayWithObject:sort];  
// 设置条件过滤(搜索name中包含字符串"Itcast-1"的记录,注意:设置条件过滤时,数据库SQL语句中的%要用*来代替,所以%Itcast-1%应该写成*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"]  
}

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

4.删除数据库中的数据

// 传入需要删除的实体对象  
[context deleteObject:managedObject];  
// 将结果同步到数据库  
NSError *error = nil;  
[context save:&error];  
if (error) {  
    [NSException raise:@"删除错误" format:@"%@", [error localizedDescription]];  
}


打开CoreData的SQL语句输出开关

1.打开Product,点击EditScheme...
2.点击Arguments,在ArgumentsPassed On Launch中添加2项
1> -com.apple.CoreData.SQLDebug
2> 1




创建NSManagedObject的子类

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


选择模型文件 


选择需要创建子类的实体 


创建完毕后,多了2个子类 


文件内容展示:
Person.h

#import <Foundation/Foundation.h>  
#import <CoreData/CoreData.h>  
  
@class Card;  
  
@interface Person : NSManagedObject  
  
@property (nonatomic, retain) NSString * name;  
@property (nonatomic, retain) NSNumber * age;  
@property (nonatomic, retain) Card *card;  
  
@end


Person.m

#import "Person.h"  
  
@implementation Person  
  
@dynamic name;  
@dynamic age;  
@dynamic card;  
  
@end


Card.h

#import <Foundation/Foundation.h>  
#import <CoreData/CoreData.h>  
  
@class Person;  
  
@interface Card : NSManagedObject  
  
@property (nonatomic, retain) NSString * no;  
@property (nonatomic, retain) Person *person;  
  
@end

Card.m

#import "Card.h"  
#import "Person.h"  
  
@implementation Card  
  
@dynamic no;  
@dynamic person;  
  
@end


那么往数据库中添加数据的时候就应该写了:

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;  
// 最后调用[context save&error];保存数据


说到这里,整个Core Data框架的入门就结束了,其实Core Data还远不止这些功能,它还支持自动撤销机制,一对多关联等,这里就不一一介绍了


本文转载自:http://blog.csdn.net/q199109106q/article/details/8563438

独-奏

独-奏

粉丝 15
博文 37
码字总数 31176
作品 6
蓬莱
程序员
私信 提问
MagicalRecord入门教程

与Core Data相关的还有比较有名的MagicalRecord。使用CoreData时会用到许多引用代码,让工作变得很复杂,MagicalRecord可以帮你解决这些问题。MagicalRecord的目标是清除CoreData引用的代码,...

Im刘亚芳
2015/02/09
0
0
服务器性能测试--Ngrinder的简介

Ngrinder入门 安装 基础环境 以 Ngrinder-3.2.3 为例,建议用 JDK-1.6: mkdir -pv /data/{app,log} wget -P /data/log dl.higkoo.com/{jdk1.6.0_45.tgz,ngrinder-controller-3.2.3-with-tom......

Bony
2016/12/18
8
0
Core Data入门

Core Data入门 简介 Core Data是iOS5之后才出现的一个框架,它提供了对象-关系映射(ORM)的功能,即能够将OC对象转化成数据,保存在SQLite数据库文件中,也能够将保存在数据库中的数据还原成O...

法斗斗
2016/03/08
9
0
solr搜索之入门及原理(一)

solr搜索技术 系列文章: solr搜索之入门及原理(一) solr搜索之demo和集成IKAnalyzer(二) solr搜索之solrJ语法(三) solr搜索之mysql导入数据到solr(四) solr搜索之tomcat运行solr(五...

wyait
2017/07/05
0
0
.NET Core实战项目之CMS 第一章 入门篇-开篇及总体规划

作者:依乐祝 原文地址:https://www.cnblogs.com/yilezhu/p/9977862.html 写在前面 千呼万唤始出来,首先,请允许我长吸一口气!真没想到一份来自28岁老程序员的自白 这篇文章会这么火,更没...

依乐祝
2018/11/18
0
0

没有更多内容

加载失败,请刷新页面

加载更多

docker中部署的应用,获取含有中文字符的验证码图片时无法正常显示

使用docker过程中遇过的最诡异的问题,服务在本地环境中,通过在IDEA里面运行,或者使用java -jar ***.war运行,获取验证码图片都没有问题,但是运行在docker中,图片正常返回,但是上面的汉...

莫在全
16分钟前
1
0
postgres+socket.io+nodejs实时地图应用实践

nodejs一直以异步io著称,其语言特性尤其擅长于在realtime应用中,如聊天室等。在进行实时应用开发时,必不可少的需要用到 socket.io库,可以说,nodejs+socket.io在实时应用中具有较好的表现...

dragon_tech
22分钟前
2
0
Java开发面试题汇总

目前流行的开发技术、常见的面试问题以及问题的答案都已经写的特别清楚了,今天我在之前的基础上,再基于个人的经验继续精选一些面试题给大家阅读参考。 1,Java的反射 Java 反射机制是在运行...

花漾年华
26分钟前
3
0
聊聊flink jdbc的ParameterValuesProvider

序 本文主要研究一下flink jdbc的ParameterValuesProvider ParameterValuesProvider flink-jdbc_2.11-1.8.0-sources.jar!/org/apache/flink/api/java/io/jdbc/split/ParameterValuesProvide......

go4it
27分钟前
1
0
UserInputControls用户输入控制

enum UserInputControls { kGovernedByOrthoMode = 0x0001,//正交模式管理 kNullResponseAccepted = 0x0002,//允许输入空 kDontEchoCancelForCtrlC = 0x0004,//ctrl C 模式不能重复......

一个小妞
48分钟前
2
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部