Core Data-备用
Core Data是一个功能强大的层,位于SQLite数据库之上,它避免了SQL的复杂性,能让我们以更自然的方式与数据库进行交互。Core Data将数据库行转换为OC对象(托管对象)来实现,这样无需任何SQL知识就能操作他们。
Core Data位于MVC设计模式中的模型层,一般需要在设备上存储结构化数据时,考虑使用SQLite或是序列化等方法,而Core Data是这两种方法的混合体,并增加了一些功能,提供了SQL强大威力,但是用起来又和序列化一样简单。Core Data能将应用程序中的对象直接保存到数据库中,无需进行复杂的查询,也无需确保对象的属性名和数据库字段名对应,这一切都由Core Data完成。
Core Data的核心——托管对象
托管对象是要存储到数据库中的对象的一种表示,可以看成是SQL记录,它通常包含一些字段,这些字段与应用程序中要存储的对象的属性进行匹配,创建托管对象后,必须将气托管到托管对象上下文中,然后才可以存储到数据库中。
托管对象上下文:
托管对象上下文包含所有的托管对象,这些托管对象已经为提交给数据库准备就绪,在托管对象上下文中,可以添加、修改和删除托管对象,这一层相当于应用程序和数据库之间的缓冲区。
托管对象表:
托管对象表描述了数据库的架构(schema),供托管对象上下文与数据库交互时使用。托管对象表包含一些列实体描述,每个实体都描述了一个数据库表,用于将托管对象映射到数据库条目。
下面来创建一个Core Data
首先要保证引入了CoreData.framwork框架到项目中,然后新建模型文件,New File——Core Data中的Data Model,然后命名为CDJournal.Xcdatamodel,这里我们做一个简单的记录流水账的程序。
接下来是定义数据库实体,选中CDJournal.Xcdatamodel文件打开表编辑器,点击添加一个名为Entry的实体,然后可以为实体添加属性并指定属性的数据类型。还可以创建其他实体,如果一个实体包含另一个实体,可通过拖放建立关系,类似于SQL外键,比如建立一个Author实体可以有多个Entry。建立实体及属性如下图:
创建完实体后必须生成表示数据库对象的类,使我们能在代码中表示实体,选择Entry实体,然后选择菜单Editor——Create NSManagedObject Subclass,点击create,就完成了。完成后可以看到工程中多了一个Entry的h和m文件,这就是Core Data模型中的实体对象。基本准备工作就完成了,如下是工程目录:
现在开始编写初始化Core Data模型的代码
首先,在AppDelegate.h中声明Core Data需要的对象,代码如下:
- #import <UIKit/UIKit.h>
- //引入CoreData框架
- #import <CoreData/CoreData.h>
- @classViewController;
- @interface AppDelegate : UIResponder <UIApplicationDelegate]]>
- @property (strong, nonatomic) UIWindow *window;
- @property (strong, nonatomic) ViewController *viewController;
- //数据模型对象
- @property(strong,nonatomic) NSManagedObjectModel *managedObjectModel;
- //上下文对象
- @property(strong,nonatomic) NSManagedObjectContext *managedObjectContext;
- //持久性存储区
- @property(strong,nonatomic) NSPersistentStoreCoordinator *persistentStoreCoordinator;
- //初始化Core Data使用的数据库
- -(NSPersistentStoreCoordinator *)persistentStoreCoordinator;
- //managedObjectModel的初始化赋值函数
- -(NSManagedObjectModel *)managedObjectModel;
- //managedObjectContext的初始化赋值函数
- -(NSManagedObjectContext *)managedObjectContext;
- @end
然后在.m文件中实现定义的方法:
- -(NSManagedObjectModel *)managedObjectModel
- {
- if (managedObjectModel != nil) {
- returnmanagedObjectModel;
- }
- managedObjectModel = [[NSManagedObjectModel mergedModelFromBundles:nil] retain];
- return managedObjectModel;
- }
- -(NSPersistentStoreCoordinator *)persistentStoreCoordinator
- {
- if (persistentStoreCoordinator != nil) {
- returnpersistentStoreCoordinator;
- }
- //得到数据库的路径
- NSString *docs = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];
- //CoreData是建立在SQLite之上的,数据库名称需与Xcdatamodel文件同名
- NSURL *storeUrl = [NSURL fileURLWithPath:[docs stringByAppendingPathComponent:@"CDJournal.sqlite"]];
- NSError *error = nil;
- persistentStoreCoordinator = [[NSPersistentStoreCoordinator alloc]initWithManagedObjectModel:[self managedObjectModel]];
- if (![persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:storeUrl options:nil error:&error]) {
- NSLog(@"Error: %@,%@",error,[error userInfo]);
- }
- returnpersistentStoreCoordinator;
- }
- -(NSManagedObjectContext *)managedObjectContext
- {
- if (managedObjectContext != nil) {
- return managedObjectContext;
- }
- NSPersistentStoreCoordinator *coordinator =[self persistentStoreCoordinator];
- if (coordinator != nil) {
- managedObjectContext = [[NSManagedObjectContext alloc]init];
- [managedObjectContext setPersistentStoreCoordinator:coordinator];
- }
- return managedObjectContext;
- }
另外,为了保证需要存储的数据不丢失,添加如下代码:
- //这个方法定义的是当应用程序退到后台时将执行的方法,按下home键执行(通知中心来调度)
- //实现此方法的目的是将托管对象上下文存储到数据存储区,防止程序退出时有未保存的数据
- - (void)applicationWillTerminate:(UIApplication *)application
- {
- NSError *error;
- if (managedObjectContext != nil) {
- //hasChanges方法是检查是否有未保存的上下文更改,如果有,则执行save方法保存上下文
- if ([managedObjectContext hasChanges] && ![managedObjectContext save:&error]) {
- NSLog(@"Error: %@,%@",error,[error userInfo]);
- abort();
- }
- }
- }
然后对xib文件进行布局并连接IBOutlet和IBAction
ViewController.h代码如下:
- #import <UIKit/UIKit.h>
- #import "AppDelegate.h"
- @interface ViewController : UIViewController
- @property (retain, nonatomic) IBOutletUITextField *titleTextField;
- @property (retain, nonatomic) IBOutletUITextField *contentTextField;
- @property (strong,nonatomic) AppDelegate *myDelegate;
- @property (strong,nonatomic) NSMutableArray *entries;
- //单击按钮后执行数据保存操作
- - (IBAction)addToDB:(id)sender;
- //单击按钮后执行查询操作
- - (IBAction)queryFromDB:(id)sender;
- //当通过键盘在UITextField中输入完毕后,点击屏幕空白区域关闭键盘的操作
- -(IBAction)backgroundTapped:(id)sender;
- @end
ViewController.m代码如下:
- #import "ViewController.h"
- #import "Entry.h"
- @interfaceViewController ()
- @end
- @implementation ViewController
- @synthesize titleTextField;
- @synthesize contentTextField;
- @synthesize myDelegate = _myDelegate;
- @synthesize entries = _entries;
- - (void)viewDidLoad
- {
- [superviewDidLoad];
- //获取当前应用程序的委托(UIApplication sharedApplication为整个应用程序上下文)
- self.myDelegate = (AppDelegate *)[[UIApplication sharedApplication] delegate];
- }
- - (void)viewDidUnload
- {
- [selfsetTitleTextField:nil];
- [selfsetContentTextField:nil];
- [superviewDidUnload];
- // Release any retained subviews of the main view.
- }
- - (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
- {
- return (interfaceOrientation != UIInterfaceOrientationPortraitUpsideDown);
- }
- - (void)dealloc {
- [titleTextFieldrelease];
- [contentTextFieldrelease];
- [superdealloc];
- }
- //单击按钮后执行数据保存操作
- - (IBAction)addToDB:(id)sender {
- //让CoreData在上下文中创建一个新对象(托管对象)
- Entry *entry = (Entry *)[NSEntityDescription insertNewObjectForEntityForName:@"Entry" inManagedObjectContext:self.myDelegate.managedObjectContext];
- [entry setTitle:self.titleTextField.text];
- [entry setBody:self.contentTextField.text];
- [entry setCreationDate:[NSDatedate]];
- NSError *error;
- //托管对象准备好后,调用托管对象上下文的save方法将数据写入数据库
- BOOL isSaveSuccess = [self.myDelegate.managedObjectContextsave:&error];
- if (!isSaveSuccess) {
- NSLog(@"Error: %@,%@",error,[error userInfo]);
- }else {
- NSLog(@"Save successful!");
- }
- }
- //单击按钮后执行查询操作
- - (IBAction)queryFromDB:(id)sender {
- //创建取回数据请求
- NSFetchRequest *request = [[NSFetchRequest alloc] init];
- //设置要检索哪种类型的实体对象
- NSEntityDescription *entity = [NSEntityDescription entityForName:@"Entry"inManagedObjectContext:self.myDelegate.managedObjectContext];
- //设置请求实体
- [request setEntity:entity];
- //指定对结果的排序方式
- NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"creationDate"ascending:NO];
- NSArray *sortDescriptions = [[NSArray alloc]initWithObjects:sortDescriptor, nil];
- [request setSortDescriptors:sortDescriptions];
- [sortDescriptions release];
- [sortDescriptor release];
- NSError *error = nil;
- //执行获取数据请求,返回数组
- NSMutableArray *mutableFetchResult = [[self.myDelegate.managedObjectContext executeFetchRequest:request error:&error] mutableCopy];
- if (mutableFetchResult == nil) {
- NSLog(@"Error: %@,%@",error,[error userInfo]);
- }
- self.entries = mutableFetchResult;
- NSLog(@"The count of entry:%i",[self.entriescount]);
- for (Entry *entry inself.entries) {
- NSLog(@"Title:%@---Content:%@---Date:%@",entry.title,entry.body,entry.creationDate);
- }
- [mutableFetchResult release];
- [request release];
- }
- //更新操作
- -(void)updateEntry:(Entry *)entry
- {
- [entry setTitle:self.titleTextField.text];
- [entry setBody:self.contentTextField.text];
- [entry setCreationDate:[NSDatedate]];
- NSError *error;
- BOOL isUpdateSuccess = [self.myDelegate.managedObjectContextsave:&error ];
- if (!isUpdateSuccess) {
- NSLog(@"Error:%@,%@",error,[error userInfo]);
- }
- }
- //删除操作
- -(void)deleteEntry:(Entry *)entry
- {
- [self.myDelegate.managedObjectContext deleteObject:entry];
- [self.entriesremoveObject:entry];
- NSError *error;
- if (![self.myDelegate.managedObjectContext save:&error]) {
- NSLog(@"Error:%@,%@",error,[error userInfo]);
- }
- }
- //当通过键盘在UITextField中输入完毕后,点击屏幕空白区域关闭键盘的操作
- -(IBAction)backgroundTapped:(id)sender
- {
- [titleTextField resignFirstResponder];
- [contentTextField resignFirstResponder];
- }
- @end
最后运行并填入数据,点击Add后添加成功
多添加几条数据后点击Query便可以查看输出的查询结果,在命令行的输出结果如下:
以上就是对Core Data的一个简单的使用,Core Data还有很多功能,这里就不一一列举,具体的在Apple的官方文档中有详细解释。
Core Data入门
一、简介
Core Data是iOS5之后才出现的一个框架,它提供了对象-关系映射(ORM)的功能,即能够将OC对象转化成数据,保存在SQLite数据库文件中,也能够将保存在数据库中的数据还原成OC对象。在此数据操作期间,我们不需要编写任何SQL语句,这个有点类似于著名的Hibernate持久化框架,不过功能肯定是没有Hibernate强大的。简单地用下图描述下它的作用:
左边是关系模型,即数据库,数据库里面有张person表,person表里面有id、name、age三个字段,而且有2条记录;
右边是对象模型,可以看到,有2个OC对象;
利用Core Data框架,我们就可以轻松地将数据库里面的2条记录转换成2个OC对象,也可以轻松地将2个OC对象保存到数据库中,变成2条表记录,而且不用写一条SQL语句。
二、模型文件
Person实体中有:name(姓名)、age(年龄)、card(身份证)三个属性
Card实体中有:no(号码)、person(人)两个属性
接下来看看创建模型文件的过程:
1.选择模板
2.添加实体
3.添加Person的2个基本属性
4.添加Card的1个基本属性
5.建立Card和Person的关联关系
右图中的表示Card中有个Person类型的person属性,目的就是建立Card跟Person之间的一对一关联关系(建议补上这一项),在Person中加上Inverse属性后,你会发现Card中Inverse属性也自动补上了
三、了解NSManagedObject对象
2.NSManagedObject的工作模式有点类似于NSDictionary对象,通过键-值对来存取所有的实体属性
1> setValue:forKey:存储属性值(属性名为key)
四、CoreData中的核心对象
注:黑色表示类名,红色表示类里面的一个属性
开发步骤总结:
1.初始化NSManagedObjectModel对象,加载模型文件,读取app中的所有实体信息
2.初始化NSPersistentStoreCoordinator对象,添加持久化库(这里采取SQLite数据库)
3.初始化NSManagedObjectContext对象,拿到这个上下文对象操作实体,进行CRUD操作
五、代码实现
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还远不止这些功能,它还支持自动撤销机制,一对多关联等,这里就不一一介绍了
Core Data-备用的更多相关文章
- Core Data入门-备用
简介 Core Data是iOS5之后才出现的一个框架,它提供了对象-关系映射(ORM)的功能,即能够将OC对象转化成数据,保存在SQLite数据库文件中,也能够将保存在数据库中的数据还原成OC对象. ...
- Core Data数据持久性存储基础教程-备用
摘要 就像我一直说的,Core Data是iOS编程,乃至Mac编程中使用持久性数据存储的最佳方式,本质上来说,Core Data使用的就是SQLite,但是通过一系列特性避免了使用SQL的一些列的麻 ...
- Core Data (2)-备用
1.Core Data 是数据持久化存储的最佳方式 2.数据最终的存储类型可以是:SQLite数据库,XML,二进制,内存里,或自定义数据类型 在Mac OS X 10.5Leopard及以后的版本中 ...
- iOS数据持久化 -- Core Data-备用
Core Data是一个功能强大的层,位于SQLite数据库之上,它避免了SQL的复杂性,能让我们以更自然的方式与数据库进行交互.Core Data将数据库行转换为OC对象(托管对象)来实现,这样无需 ...
- iOS之Core Data及其线程安全
一.简介 Core Data是iOS5之后才出现的一个框架,它提供了对象-关系映射(ORM)的功能,即能够将OC对象转化成数据,保存在SQLite数据库文件中,也能够将保存在数据库中的数据还原成OC对 ...
- Core Data的一些常见用法
一.简介 Core Data是一个纯粹的面向对象框架,其本质就是一个ORM(对象关系映射:Object Relational Mapping),能以面向对象的方式操作SQLite数据库.在实际开发中绝 ...
- Core Data 使用映射模型
Core Data 使用映射模型 如果新版本的模型存在较复杂的更改,可以创建一个映射模型,通过该模型指定源模型如何映射到目标模型. 创建映射模型,新建File, Core Data 选择Mappin ...
- SELF, self in CORE DATA
Predicate SELF Represents the object being evaluated. CORE DATA Retrieving Specific Objects If your ...
- Core Data浅谈初级入门
Core Data是iOS5之后才出现的一个框架,它提供了对象-关系映射(ORM)的功能,即能够将OC对象转化成数据,保存在SQLite数据库文件中,也能够将保存在数据库中的数据还原成OC对象.在此数 ...
随机推荐
- HDOJ(HDU) 1994 利息计算(简单题目)
Problem Description 为自行解决学费,chx勤工俭学收入10000元以1年定期存入银行,年利率为3.7% .利率 按年计算,表示100元存1年的利息为3.7元.实际上有时提前有时推迟 ...
- android判断应用是否有某个权限
1) 判断应用是否具有某个权限 PackageManager pm = getPackageManager(); boolean permission = (PackageManager.PERMIS ...
- Scala 编程(四)内建控制结构
if 表达式 Scala 的 if 如同许多其它语言中的一样工作.它测试一个状态并据其是否为真,执行两个分支中的一个: var filename = "default.txt" i ...
- java笔记12之面向对象初始
1 概述 类:是一组相关的属性和行为的集合.是一个抽象的概念. 对象:是该类事物的具体表现形式.具体存在的个体. (1)面向对象思想 面向对象是基于面向过程的编程思想. ...
- web端及时通讯原理
前言 有关IM(InstantMessaging)聊天应用(如:微信,QQ).消息推送技术(如:现今移动端APP标配的消息推送模块)等即时通讯应用场景下,大多数都是桌面应用程序或者native应用较为 ...
- 我与Ubuntu的5年共成长
初次接触Ubuntu那还是2010年的初夏,那年大学二年级即将结束,为了增加计算机学院学生的技术能力和就业竞争力,学校组织了很多培训机构.公司企业来学校做技术宣讲.分享等 记得有一个企业是做Mp3 M ...
- android系统的日志查看
日志跟踪(alog) 在我们开发手机软件的过程中,常常需要通过USB线连接电脑看日志.但在windows下,我们要安装驱动.有的android板子,就不支持USB调试.所以我开发了这个小工具.方便打开 ...
- Python使用mechanize模拟浏览器
Python使用mechanize模拟浏览器 之前我使用自带的urllib2模拟浏览器去进行訪问网页等操作,非常多站点都会出错误,还会返回乱码.之后使用了 mechanize模拟浏览器,这些情况都没出 ...
- VS2008 动态库和静态库的生成和加载
第一:动态库和静态库的生成: 1) 新建一个生成dll工程: 文件->新建->项目->Win32->Win32控制台应用程序 输入项目名称:dllTest ,项目路径:D:\V ...
- 使用Intent实现Activity的显式跳转
[正文] 这里以按钮实现活动跳转为例,为实现这个功能,我们需要三个步骤: 1.点击按钮才发生页面跳转,因此,第一步我们先要找到要点击的按钮 如何拿到按钮对象呢?通过资源id,前面我们提到过,在R.id ...