CoreData学习-最好的一片文章

分类: IOS重新上路2014-05-25 18:00 1937人阅读 评论(0) 收藏 举报

目录(?)[+]

写的很好的一篇教程,我什么时候能写出这么棒的文章来,还是继续努力学习吧。

iOS教程:Core Data数据持久性存储基础教程

其实最近更多的是在写这篇文章《iOS教程:使用持久性数据Core Data》,这篇是《iOS开发教程:Storyboard全解析-第一部分》这篇的后续,但是目前还没有完成,先放出一个持久性数据存储的教程以供参考。这其实是一篇翻译文章,英文的原文见这里。我翻译的过程中改变了一些内容以便适应我们中国人的口味,下面请看教程:

就像我一直说的,Core Data是iOS编程,乃至Mac编程中使用持久性数据存储的最佳方式,本质上来说,Core Data使用的就是SQLite,但是通过一系列特性避免了使用SQL的一些列的麻烦,不仅如此,他还能够合理管理内存,反正好处很多,我们推荐使用。

在这个教程中,我们将会创建一个Core Data的可视模型,之后再做一个Table View,让Table View的内容能够存储在数据模型里。

创建Core Data工程

首先打开Xcode,新建一个工程,选择Master-Detail模板。

我们这里使用FailedBankCD作为工程名称。

记住一定要选中Use StoryboardsUse Core Data, and Use Automatic Reference Counting 这几个选项,之后创建。

在开始之前,我想先把一些没有用的模板文件删除掉,选中一下四个文件。

  • FBCDMasterViewController.h
  • FBCDMasterViewController.m
  • FBCDDetailViewController.h
  • FBCDDetailViewController.m

删除之,一了百了,选择 “Move to Trash”。

现在使用Obj-c class的模板新建一个文件,命名为FBCDMasterViewController 他是个UITableViewController,记住下面的几个钩都不能选。

选中FBCDMasterViewController.h 在@end 之前的结尾行加入代码:

@property (nonatomic,strong) NSManagedObjectContext* managedObjectContext;

现在来到 .m 文件, 加入下面的语句。

@synthesize managedObjectContext;

如果你不知道什么是 “NSManagedObjectContext” 那也没关系,我们一会之后会谈这问题。

但是我们还是得先在Storyboard中删除掉我们刚才所删除的类所对应的视图,看图:

最后,选中FailedBanksCD.xcdatamodel,你会看到出现了一个可视编辑器我们接下来就用这个编辑器来编辑我们的数据模型,选中中间显示“Entity”(实体)的小泡泡,之后删除之。

如果你的编辑器看起来和下面的不一样,那么请将编辑器风格(Editor Style)改成visual view

OK,大功告成,如果现在启动这个App,你会发现他就是个空白的应用。

打开 FailedBanksCDAppDelegate.m,你会看到已经有一些预置的函数在这里了,这是为了建立Core Data的栈,包括如何创建数据模型,如何管理数据模型,如何建立持久性数据协调器等等。

如果这些术语看的你头疼,那也没关系,看了下面的解释你就会明白了:

  • Managed Object Model(管理数据模型): 你可以将这个东西看作是数据库的轮廓,或者结构。这里包含了各个实体的定义信息,一般来说,你会使用我们刚刚看过的视觉编辑器来操作这个物体,添加属性,建立属性之间的关系等等,当然你也可以使用代码。
  • Persistent Store Coordinator (持久性数据协调器): 你可以将这个东西看作是数据库连接库,在这里,你将设置数据存储的名字和位置,以及数据存储的时机。
  • Managed Object Context (管理数据内容):你可以将这一部分看作是数据的实际内容,这也是整个数据库中对我们而言最重要的部分(这还用说),基本上,插入数据,查询数据,删除数据的工作都在这里完成。
现在你还不用十分了解这些方法,因为我们暂时还用不到,但是了解一些概念总是好的。

创建数据模型

Core Data与SQLite不同的是,你不能够提取一个实体的某些属性,你只能够把整个实体提取出来,,之后再将他们分解。

让我们看看这是如何运作的,打开视觉编辑器(单击FailedBanksCD.xcodedatamode)

在底部的工具栏中,单机加号,新建一个实体。

新建实体之后,视觉编辑器会显示他的属性

将这个新实体命名为FailedBankInfo,之后单机这个实体,确保可视部分的第三个标签都是被选中的。在属性检查器的第三个标签里,你就会看到这个实体是NSManagedObject的子类,这是实体的默认类,我们现在先用着,将来我们会返回来修改为一个自定义的类。

现在,让我们加入一些属性吧(Atrributes),确保你的实体是被选中的状态,之后在中间的面板的底部按下加号,之后会出现一个像下面这样的选项框。

在Data Model属性检查器中,向下面那样将这个属性命名为“name”,type设置为“String”

 

现在,再继续增加两个新的属性“city”和“State”,类型都是字符串。

接下来,我们创建一个名为FailedBankDetails的实体,你一定已经会弄了吧?之后增加下面的属性在里面,zip(integer32属性)closeDate(date属性)updateDate(date属性)。

最后,我们将这两种实体连接起来,选择FailedBankInfo,按住中间面板的加号键不放,选择 “Add relationship”:

 

将这个relationship命名为“details”, 将目标设置为 “FailedBankDetails.”

好了,我们刚刚设置了一个连接两个实体的关系,这意味着,每一个FailedBankInfo的属性都将会拥有一个一个FailedBankDetails的实体,在设置的背后,Core Data会自动设置FailedBankInfo中实体的ID,不过这些我们不需要了解。

Apple官方建议说,每当你建立一个目标关系时,最好建立一个返回的关系,所以我们就按照官方的指示做吧。

在“FailedBankDetails” 中建立一个叫做 “info”的关系,设置他的目标是  “FailedBankInfo”,正好与“Details”这个关系是反着的。

接着,我们设置这两个关系的删除规则为“cascade”,这意味着,如果你删除了其中一个数据,另一个实体中的数据也会跟着本删除。

运行一下,怎么?崩溃了??

哦,原来是之前已经存在的一个数据已经无法被我们修改过的数据模型读取了,在这种情况下,只需要删除模拟器中的app,再重新运行就可以了。

测试我们的数据模型

不管你怎么想,测试可是我们编辑数据库中最重要的一步。

首先,我们向我们的数据库中加入一些测试数据,打开FailedBanksCDAppDelegate.m,在顶部application:didFinishLaunchingWithOptions方法中加入下面的代码:

NSManagedObjectContext *context = [self managedObjectContext];
NSManagedObject *failedBankInfo = [NSEntityDescriptioninsertNewObjectForEntityForName:@"FailedBankInfo"inManagedObjectContext:context];
[failedBankInfo setValue:@"Test Bank" forKey:@"name"];
[failedBankInfo setValue:@"Testville" forKey:@"city"];
[failedBankInfo setValue:@"Testland" forKey:@"state"];
NSManagedObject *failedBankDetails = [NSEntityDescriptioninsertNewObjectForEntityForName:@"FailedBankDetails"inManagedObjectContext:context];
[failedBankDetails setValue:[NSDate date] forKey:@"closeDate"];
[failedBankDetails setValue:[NSDate date] forKey:@"updateDate"];
[failedBankDetails setValue:[NSNumber numberWithInt:12345] forKey:@"zip"];
[failedBankDetails setValue:failedBankInfo forKey:@"info"];
[failedBankInfo setValue:failedBankDetails forKey:@"details"];
NSError *error;
if (![context save:&error]) {NSLog(@"Whoops, couldn't save: %@", [error localizedDescription]);
}

在第一行,我们创建了一个指向我们的数据库的指针。

接着,我们为FailedBankInfo实体创建一个NSManagedObject实体,这里使用的是insertNewObjectForName的方法,每一个使用Core Data储存数据的方法都是由NSManagedObject中衍生出来的,当你创建了这个方法的实例的时候,你就可以给我们刚才在视觉编辑器中创建的模型中的任何属性进行赋值了。

之后我们设置一个测试数据,在这里数据只在内存中被修改,要是想要存入数据库,我们必须使用managedObjectContext方法

插入一个数据就是这么简单,完全不用SQL语句。

在运行之前,我们先用一些代码来列出我们数据库中的数据。

NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
NSEntityDescription *entity = [NSEntityDescriptionentityForName:@"FailedBankInfo" inManagedObjectContext:context];
[fetchRequest setEntity:entity];
NSArray *fetchedObjects = [context executeFetchRequest:fetchRequest error:&error];
for (NSManagedObject *info in fetchedObjects) {NSLog(@"Name: %@", [info valueForKey:@"name"]);NSManagedObject *details = [info valueForKey:@"details"];NSLog(@"Zip: %@", [details valueForKey:@"zip"]);
}

这里我们创建一个叫做fetch request的新方法,你可以将一个fetch request看做Sql中的select语句,我们调用entityForName方法来获取一个指向FailedBankInfo的指针,之后使用setEntity方法来告诉我们的fetch request我们想要的是哪一种的实体。

之后,我们调用executeFetchRequest方法,将FailedBankInfo表中的所有数据推入一个数据缓存中,之后枚举所有NSManagedObject,使用valueForKey语句来调用其中的数据。

注意尽管我们从FailedBankInfo表中推出了所有的数据,我们仍然可以通过FailedBankInfo实体中的详细数据来访问FailedBankDetails这个数据体。

运行一下这个应用,目前你在屏幕上什么也看不到,你可以看看程序的输出窗口,你就可以看到输出结果了,每次你设置数据库的时候都应该做这样一个测试。

来看看SQL语句的真面目

我不知道你怎么想的,但是我个人喜欢看到每个语句后面的SQL语句,以确定这个程序正在按照我想的方式前进。

Apple提供了一个这样做的简便的方法,看下图,在Edit Scheme中选择Run,之后进入Arguments标签,加入下面的语句:“-com.apple.CoreData.SQLDebug 1”,完成之后,你会看到第二张图:

现在,每当你运行这个程序,Debug栏就会输出正在进行的活动的SQL语句了。

如果你对SQL语句不了解,也没有关系,没有人强制你学习这个烦人的东西,你可以一直使用Core Data。

自动生成的模型文件

目前为止,我们一直在用 NSManagedObject 来处理我们的实体,这并不是最好的方式,为什么呢?因为NSManagedObject不是一个 强型类(strongly typed class),所以,你只能够使用字符串来访问数据属性,如果打错了的话,就会造成错误,很不爽。

更好的方法是为每一个实体都创建一个模型文件,这样做好处很多,不赘述,Xcode提供了一个类生成器让我们方便地完成这个任务。

来试试看,打开 FailedBanksCD.xcdatamodel,点击 FailedBankInfo entity,进入File——NewFile。选择Core DataNSManagedObject 模板,之后创建:

 

你应该可以看到,你的工程中添加了一些新的文件,就是FailedBankInfo.h/m 和 FailedBankDetails.h/m,这些是一些非常简单的类,只是为了声明你在实体中添加的属性而已,Xcode会动态的添加语句进去,我们不用管这些。

为FailedBankDetails 实体做同样的事情。

但是这些自动生成的类中有一个问题我们必须手动解决,如果你打开FailedBankDetails.h来看,你就会发现info变量被声明称了FailedBankInfo,但是在FailedBankInfo.h中,他被声明称了NSManagedObject类,其实应该是FailedBankDetails类。

这是因为FailedBankInfo的生成器在FailedBankDetails的生成器之前运行,所以生成器不知道后面还有个这玩意。

你可以像上面那样手动修复,但是也有简单的方法,就是重新运行FailedBankInfo的生成器。

如果我们回到FailedBanksCD.xcdatamodel,当你查看实体的类的时候,你会发现他们已经被自动修改为自动创建的类。

为了测试,打开BCDAppDelegate.m ,在头部加入:

#import "FailedBankInfo.h"
#import "FailedBankDetails.h"

之后修改代码:

NSManagedObjectContext *context = [self managedObjectContext];
FailedBankInfo *failedBankInfo = [NSEntityDescriptioninsertNewObjectForEntityForName:@"FailedBankInfo"inManagedObjectContext:context];
failedBankInfo.name = @"Test Bank";
failedBankInfo.city = @"Testville";
failedBankInfo.state = @"Testland";
FailedBankDetails *failedBankDetails = [NSEntityDescriptioninsertNewObjectForEntityForName:@"FailedBankDetails"inManagedObjectContext:context];
failedBankDetails.closeDate = [NSDate date];
failedBankDetails.updateDate = [NSDate date];
failedBankDetails.zip = [NSNumber numberWithInt:12345];
failedBankDetails.info = failedBankInfo;
failedBankInfo.details = failedBankDetails;
NSError *error;
if (![context save:&error]) {NSLog(@"Whoops, couldn't save: %@", [error localizedDescription]);
}// Test listing all FailedBankInfos from the store
NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"FailedBankInfo"inManagedObjectContext:context];
[fetchRequest setEntity:entity];
NSArray *fetchedObjects = [context executeFetchRequest:fetchRequest error:&error];
for (FailedBankInfo *info in fetchedObjects) {NSLog(@"Name: %@", info.name);FailedBankDetails *details = info.details;NSLog(@"Zip: %@", details.zip);
}

这和我们上次测试的代码差不多,是不是?

创建一个表视图

打开 FBCDMasterViewController.h 声明一个数组,你懂得(如果不懂请见《如何创建一个简单的表》)

@property (nonatomic, strong) NSArray *failedBankInfos;

请注意,我们在测试的时候已经使用了一些方法来获取数据了,就在FBCDAppDelegate.m中的application:didFinishLaunchingWithOptions方法中,记得吗?

回到FailedBanksListViewController.m 作出以下修改:

// At very top, in import section
#import "FailedBankInfo.h"// At top, under @implementation
@synthesize failedBankInfos;

修改 viewDidLoad 方法:

- (void)viewDidLoad {[super viewDidLoad];NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];NSEntityDescription *entity = [NSEntityDescriptionentityForName:@"FailedBankInfo" inManagedObjectContext:managedObjectContext];[fetchRequest setEntity:entity];NSError *error;self.failedBankInfos = [managedObjectContext executeFetchRequest:fetchRequest error:&error];self.title = @"Failed Banks";}

这和我们测试的代码差不多是不是?我们使用一个 fetch request 来获取数据库中的数据,之后储存在内存中。

numberOfSectionsInTableView:

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {return 1;
}

numberOfRowsInSection:

- (NSInteger)tableView:(UITableView *)tableViewnumberOfRowsInSection:(NSInteger)section {return [failedBankInfos count];
}

cellForRowAtIndexPath :

- (UITableViewCell *)tableView:(UITableView *)tableViewcellForRowAtIndexPath:(NSIndexPath *)indexPath {static NSString *CellIdentifier = @"Cell";UITableViewCell *cell =[tableView dequeueReusableCellWithIdentifier:CellIdentifier];// Set up the cell...FailedBankInfo *info = [failedBankInfos objectAtIndex:indexPath.row];cell.textLabel.text = info.name;cell.detailTextLabel.text = [NSString stringWithFormat:@"%@, %@",info.city, info.state];return cell;
}

以上是制作一个表视图的代码,很熟悉吧,接下来我们在Storyboard中制作这个表视图,记得将cell的style设置为subtitle。

运行一下这个App,看看吧。

之后看些什么?

这是我制作完成的例子程序源码,欢迎下载。

这是原作者的样板程序: sample code for the project so far (direct download).

此条目是由 foreveryh 发表在 iPhone开发、SDK 分类目录的。将固定链接加入收藏夹。

  • 上一篇

转载于:https://www.cnblogs.com/iOS-mt/p/4261475.html

CoreData学习-最好的一片文章相关推荐

  1. 【每周CV论文推荐】 深度学习人脸检测入门必读文章

    欢迎来到<每周CV论文推荐>.在这个专栏里,还是本着有三AI一贯的原则,专注于让大家能够系统性完成学习,所以我们推荐的文章也必定是同一主题的. 人脸图像是整个图像处理领域里面研究时间最长, ...

  2. 深度学习的150多篇文章和10多个专栏推荐

    文章首发于微信公众号<有三AI> 创业第一天,有三AI扔出了深度学习的150多篇文章和10多个专栏 文/编辑 | 言有三 在这篇文章中,有三跟大家来聊一下有三AI和如何学习深度学习这件事儿 ...

  3. fasterrcnn论文_【每周CV论文推荐】 深度学习人脸检测入门必读文章

    我的新书<深度学习之人脸图像算法>市了,欢迎大家关注! 言有三新书来袭,业界首次深入全面讲解深度学习人脸图像算法​mp.weixin.qq.com 欢迎来到<每周CV论文推荐> ...

  4. 学习,教育的1000+篇文章总结

    学习,教育的1000+篇文章总结 本文收集和总结了有关学习,教育的1000+篇文章,由于篇幅有限只能总结近期的内容,想了解更多内容可以访问:http://www.ai2news.com/, 其分享了有 ...

  5. 【小样本学习】什么是小样本学习?这篇综述文章用166篇参考文献告诉你答案...

    点击上方"AI遇见机器学习",选择"星标"公众号 重磅干货,第一时间送达 来源:机器之心 再次编辑:夕小瑶的卖萌屋 什么是小样本学习?它与弱监督学习等问题有何差 ...

  6. 「每周CV论文推荐」 初学深度学习人脸关键点检测必读文章

    人脸关键点检测是人脸图像中重要的基石,今天给大家介绍入门深度学习人脸关键点检测必读的文章. 作者&编辑 | 言有三 1 DCNN Cascade 听这个名字就知道是一个很早期的,使用Casca ...

  7. 【每周CV论文推荐】 初学深度学习人脸关键点检测必读文章

    欢迎来到<每周CV论文推荐>.在这个专栏里,还是本着有三AI一贯的原则,专注于让大家能够系统性完成学习,所以我们推荐的文章也必定是同一主题的. 人脸关键点检测是人脸图像中重要的基石,今天给 ...

  8. 创业第一天,有三AI扔出了深度学习的150多篇文章和10多个专栏

    文/编辑 | 言有三 在这篇文章中,有三跟大家来聊一下有三AI和如何学习深度学习这件事儿. 1 概述 自我介绍 "有三AI"创始人网名言有三,本名龙鹏,本科就读于华中科技大学,硕士 ...

  9. 如何学习数据结构和算法——大佬文章汇总

    第一篇 第二篇. 作者:左程云 我分别说一下国内和国外的行情. 国内的话,一般来讲,工资高的公司在面试时算法和数据结构题目的比重较大,工资一般的公司比重较小.当然同样公司的不同岗位,要求也会不同,但总 ...

  10. 什么是小样本学习?这篇综述文章用166篇参考文献告诉你答案

    一只小狐狸带你解锁 炼丹术&NLP 秘籍 来源:机器之心 什么是小样本学习?它与弱监督学习等问题有何差异?其核心问题是什么?来自港科大和第四范式的这篇综述论文提供了解答. 数据是机器学习领域的 ...

最新文章

  1. k8s使用kube-router网络插件并监控流量状态
  2. java 右键获取路径,java路径
  3. 3 镜像仓库Harbor安装
  4. Linux下多线程查看工具(pstree、ps、pstack),linux命令之-pstree使用说明
  5. MySQL查询对NULL的处理
  6. linux下代码写错了怎么更改_AWS全面上市开源Linux发行版,专为容器操作系统设计...
  7. 基于token与基于服务器的身份认证
  8. C#LeetCode刷题之#551-学生出勤纪录 I​​​​​​​(Student Attendance Record I)
  9. 阿里RDS开发专家解析MySQL各版本并行复制
  10. map转成url拼接请求参数
  11. hibernate反向工程
  12. ANTLR实现的SQL解析器 - OQL
  13. C字符串指针遇到的问题
  14. vim插件管理器:Vundle的介绍及安装
  15. 天涯 大神 kkndme 房地产 调控
  16. welearn综合教程网课答案
  17. 2021最新申请苹果的公司开发者账号
  18. 记录一下第一个个人小程序
  19. elasticsearch基础知识
  20. 【使用C++开发MCU】05-CAN实例之NXP S32K1 FlexCAN模块

热门文章

  1. java笔记之过滤器
  2. Linux中eclipse配置Maven,eclipse maven选项怎么配置settings
  3. qt messagebox退出程序_Qt多线程创建
  4. c语言的加法和平均值程序,编写求一组整数的和与平均值的程序
  5. linux还原系统_Linux Kernel 5.5 最终删除 SYSCTL 系统调用
  6. python类的属性前加上双下划线_Python中使用双下划线防止类属性被覆盖问题
  7. Spring源码之bean的加载(五)准备创建bean
  8. 【渝粤教育】国家开放大学2018年秋季 0716-21T工程建设法规 参考试题
  9. Pandas系列(七)Pandas数据排序
  10. 【重磅开源】Facebook开源 Nevergrad:一种用于无梯度优化的开源工具