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需要的对象,代码如下:
Cpp代码
- #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文件中实现定义的方法:
Cpp代码
- -(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;
- }
另外,为了保证需要存储的数据不丢失,添加如下代码:
Cpp代码
- //这个方法定义的是当应用程序退到后台时将执行的方法,按下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代码如下:
Cpp代码
- #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代码如下:
Cpp代码
- #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语句。
二、模型文件
在Core Data,需要进行映射的对象称为实体(entity),而且需要使用Core Data的模型文件来描述app中的所有实体和实体属性。这里以Person(人)和Card(身份证)2个实体为例子,先看看实体属性和实体之间的关联关系:
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对象
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.搭建上下文环境
[java] view plaincopy
- // 从应用程序包中加载模型文件
- 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.添加数据到数据库
[java] view plaincopy
- // 传入上下文,创建一个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.从数据库中查询数据
[java] view plaincopy
- // 初始化一个查询请求
- 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.删除数据库中的数据
[java] view plaincopy
- // 传入需要删除的实体对象
- [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
[java] view plaincopy
- #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
[java] view plaincopy
- #import "Person.h"
- @implementation Person
- @dynamic name;
- @dynamic age;
- @dynamic card;
- @end
Card.h
[java] view plaincopy
- #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
[java] view plaincopy
- #import "Card.h"
- #import "Person.h"
- @implementation Card
- @dynamic no;
- @dynamic person;
- @end
那么往数据库中添加数据的时候就应该写了:
[java] view plaincopy
- 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还远不止这些功能,它还支持自动撤销机制,一对多关联等,这里就不一一介绍了