如何使用Core Data

如何使用Core Data,第1张

第一步:创建一个 DataDemo.xcdatamodel 文件,在其中建一个实体名字为 Entity1 ,实体中建两个字段 id 和 name。

第二步:为项目添加 CoreData.framework 框架,然后在 .pch 中加入 #import <coredata/coredata.h>

第三步:在 app delegate 中定义 core data 的相关对象(同时完成它们的初始化工作)

.h 文件:

#import <coredata/coredata.h>

@private

NSManagedObjectModel *managedObjectModel

NSManagedObjectContext *managedObjectContext

NSPersistentStoreCoordinator *persistentStoreCoordinator

@property (nonatomic, retain, readonly) NSManagedObjectModel *managedObjectModel

@property (nonatomic, retain, readonly) NSManagedObjectContext *managedObjectContext

@property (nonatomic, retain, readonly) NSPersistentStoreCoordinator *persistentStoreCoordinator

- (NSString*) applicationDocumentsDirectory

- (void) saveContext

.m 文件:

@synthesize managedObjectModel

@synthesize managedObjectContext

@synthesize persistentStoreCoordinator

- (void)dealloc {

[managedObjectModel release]

[managedObjectContext release]

[persistentStoreCoordinator release]

[super dealloc]

}

- (NSString*)applicationDocumentsDirectory

{

//return [[[NSFileManager defaultManager] URLsForDirectory:NSDocumentDirectory inDomains:NSUserDomainMask] lastObject]

NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)

NSString *basePath = ([paths count] >0) ? [paths objectAtIndex:0] : nil

return basePath

//return [NSURL URLWithString:basePath]

}

- (void)saveContext{

NSError *error = nil

NSManagedObjectContext *objectContext = self.managedObjectContext

if (objectContext != nil)

{

if ([objectContext hasChanges] &&![objectContext save:&error])

{

NSLog(@"Unresolved error %@, %@", error, [error userInfo])

abort()

}

}

}

- (NSManagedObjectContext *)managedObjectContext

{

if (managedObjectContext != nil) {

return managedObjectContext

}

NSPersistentStoreCoordinator *coordinator = [self persistentStoreCoordinator]

if (coordinator != nil) {

managedObjectContext = [[NSManagedObjectContext alloc] init]

[managedObjectContext setPersistentStoreCoordinator:coordinator]

}

return managedObjectContext

}

- (NSManagedObjectModel *)managedObjectModel

{

if (managedObjectModel != nil) {

return managedObjectModel

}

//从本地所有 xcdatamodel 文件中获得这个 CoreData 的数据模板

managedObjectModel = [[NSManagedObjectModel mergedModelFromBundles:nil] retain]

return managedObjectModel

}

- (NSPersistentStoreCoordinator *)persistentStoreCoordinator

{

if (persistentStoreCoordinator != nil) {

return persistentStoreCoordinator

}

//数据库名为TestDB.sqlite

NSURL *storeUrl = [NSURL fileURLWithPath:[[self applicationDocumentsDirectory] stringByAppendingPathComponent:@"TestDB.sqlite"]]

NSError *error

persistentStoreCoordinator = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:[self managedObjectModel]]

if (![persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreType configuration:nilURL:storeUrl options:nil error:&error]) {

NSAssert(0, @"persistentStoreCoordinator init failed!")

}

return persistentStoreCoordinator

}

- (void)applicationWillTerminate:(UIApplication *)application

{

NSError *error

if (managedObjectContext != nil) {

if ([managedObjectContext hasChanges] &&![managedObjectContext save:&error]) {

NSAssert(0, @"save changes failed when terminage application!")

}

}

}

第四步:在相关的 viewcontroller 中 *** 作数据,基本的 core data *** 作代码如下:

NSError *error

testSQLiteAppDelegate *app = (testSQLiteAppDelegate*)[[UIApplication sharedApplication] delegate]

NSManagedObjectContext *context = app.managedObjectContext

//1、插入

NSManagedObject *newManagedObject = nil

newManagedObject = [NSEntityDescription insertNewObjectForEntityForName:@"Entity1" inManagedObjectContext:context]

[newManagedObject setValue:[NSNumber numberWithInt:1] forKey:@"id"]

[newManagedObject setValue:@"i love you" forKey:@"name"]

if (![context save:&error]){

NSLog (@"The error is: %@", [error userInfo])

}

else

{

NSLog (@"The newManagedObject's id is: %@", [newManagedObject valueForKey:@"id"])

}

//2、查询

NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init]

NSEntityDescription *entity = [NSEntityDescription entityForName:@"Entity1" inManagedObjectContext:context]

NSSortDescriptor *sortDescriptor1 = [[NSSortDescriptor alloc] initWithKey:@"id" ascending:YES]

NSSortDescriptor *sortDescriptor2 = [[NSSortDescriptor alloc] initWithKey:@"name" ascending:YES]

NSArray *sortDescriptors = [[NSArray alloc] initWithObjects:sortDescriptor1, sortDescriptor2, nil]

[fetchRequest setSortDescriptors:sortDescriptors]

[fetchRequest setEntity:entity]

[fetchRequest setFetchBatchSize:20]

NSArray *objects = [context executeFetchRequest: fetchRequest error:&error]

if (objects == nil)

{

NSLog(@"There was an error!")

}

else {

for (NSManagedObject *oneObject in objects)

{

NSLog(@"%@",[oneObject valueForKey:@"name"])

}

}

//3、删除

[context deleteObject:[objects objectAtIndex:0]]

if (![context save:&error]) {

exit(-1)

}

这里更详细的补充一点关于 core data 的 *** 作代码:

1、条件过滤:选择 cid=1 的数据

- (NSFetchedResultsController*) fetchedResultsController

{

NSPredicate *predicate = [NSPredicate predicateWithFormat:@"cid=1"]

[fetchRequest setPredicate:predicate]

}

2、通用代码,在 viewcontroller 里面会用到:

TestViewController.h 的代码:

@interface TestViewController : UITableViewController <NSFetchedResultsControllerDelegate>

{

NSFetchedResultsController *fetchedResultsController

NSManagedObjectContext *managedObjectContext

}

@property (nonatomic, retain) NSFetchedResultsController *fetchedResultsController

@property (nonatomic, retain) NSManagedObjectContext *managedObjectContext

- (void)configureCell:(UITableViewCell *)cell withCategory:(Category *)category

@end

TestViewController.m 的代码:

-(NSFetchedResultsController *)fetchedResultsController

{

if (fetchedResultsController != nil)

{

return fetchedResultsController

}

NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init]

NSEntityDescription *entity = [NSEntityDescription entityForName:@"Category" inManagedObjectContext:managedObjectContext]

[fetchRequest setEntity:entity]

NSSortDescriptor *nameDescriptor = [[NSSortDescriptor alloc] initWithKey:@"name" ascending:YES]

NSArray *arraySortDescriptor = [[NSArray alloc] initWithObjects:nameDescriptor, nil]

[fetchRequest setSortDescriptors:arraySortDescriptor]

NSPredicate *predicate = [NSPredicate predicateWithFormat:cidString]

[fetchRequest setPredicate:predicate]

NSFetchedResultsController *frController = [[NSFetchedResultsController alloc]

initWithFetchRequest:fetchRequest managedObjectContext:managedObjectContext sectionNameKeyPath:@"name" cacheName:@"Root"]

frController.delegate = self

self.fetchedResultsController = frController

[frController release]

[fetchRequest release]

[nameDescriptor release]

return fetchedResultsController

}

- (void)controllerWillChangeContent:(NSFetchedResultsController *)

{

[self.tableView beginUpdates]

}

- (void)controller:(NSFetchedResultsController *)controller didChangeObject:(id)anObject

atIndexPath:(NSIndexPath *)indexPath forChangeType: (NSFetchedResultsChangeType)type

newIndexPath:(NSIndexPath *)newIndexPath

{

UITableView *tableView = self.tableView

switch (type) {

case NSFetchedResultsChangeInsert:

[tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationFade]

break

case NSFetchedResultsChangeDelete:

[tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade]

break

case NSFetchedResultsChangeMove:

[tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade]

[tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationFade]

break

case NSFetchedResultsChangeUpdate:

[self configureCell:[tableView cellForRowAtIndexPath:indexPath] withCategory:anObject]

break

default:

break

}

}

- (void)controller:(NSFetchedResultsController *)controller didChangeSection:(id <NSFetchedResultsSectionInfo>)sectionInfo

atIndex:(NSUInteger)sectionIndex forChangeType:(NSFetchedResultsChangeType)type

{

switch(type) {

case NSFetchedResultsChangeInsert:

[self.tableView insertSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade]

break

case NSFetchedResultsChangeDelete:

[self.tableView deleteSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade]

break

}

}

- (void) controllerDidChangeContent:(NSFetchedResultsController *)controller

{

[self.tableView endUpdates]

}

1)先创建需要的实体表(方法见"CoreData单表创建使用")

Course, Section, Teacher 表

2)建立关联 设置删除关系,对应关系如下(设置的具体含义已在文章第二部分介绍)

课程_教师 (一课程可包含多个教师,删除关系:作废)

课程_章节(一节课可包含多个章节,删除关系:级联)

章节_课程(一个章节只属于一个课程,删除关系:作废)

教师_课程(一个教师可以讲多门课,删除关系:作废)

创建之后查看style关系:

表关系

3)使用创建的对象生成相应地文件:.h和.m(方法见"CoreData单表创建使用")

根据3个对象创建的文件

根据设置的对象之间的对应关系 会生成相应地属性和方法Section对应一个课程时 添加课程对象的属性

课程对应的章节以及教师都是多个,生成NSSet 存储 并生成相应方法

NSSet 是一个无序的集合 ,可以使用NSSet 与NSArray根据需要转换使用,其他使用方法 同单表使用的增删查改,只是访问时多层访问.

如:NSPredicate*predict = [NSPredicate predicateWithFormat:@"section_course.courseId = %d",courseId]

如:self.courseEntity.course_teacher.count

coreData多表关联的基本 *** 作已做一个简单的介绍,具体的功能还待使用者发掘,并且苹果提供了NSFetchController 与coreData结合使用.

首先,coredata和sqlite的概念不同,core为对象周期管理,而sqlite为dbms。

下面的讨论以使用core data来做数据持久化并使用sqlite做backend存储的情况为前提。

使用方便性。实际上,一个成熟的工程中一定是对数据持久化进行了封装的,因此底层使用的到底是core data还是sqlite,不应该被业务逻辑开发者关心。因此,即使习惯写SQL查询的人,也应该避免在业务逻辑中直接编写SQL语句。

存储性能,在写入性能上,因为都是使用的sqlite格式作为磁盘存储格式,因此其性能是一样的,如果你觉得用core data写的慢,很可能是你用sqlite的时候写的每条数据的内容没有core data时多,或者是你批量写入的时候每写入一条就调用了一次save。

查询性能,core data因为要兼容多种后端格式,因此查询时,其可用的语句比直接使用sqlite少,因此有些fetch实际上不是在sqlite中执行的。但这样未必会降低查询效率。因为iPhone的flash memory速度还是很快的。我的经验是大部分时候,在内存不是很紧张时,直接fetch一个entity的所有数据然后在内存中做filter往往比使用predicate在fetch时过滤更快。如果你觉的查询慢,很可能是查询方式有问题,可以把core data的debug模式打开,看一下到底执行了多少SQL语句,相信其中大部分是可以通过改写core data的调用方式避免的。

core data的一个比较大的痛点是多人合作开发的时候,管理coredata的模型需要很小心,尤其是合并的时候,他的data model是XML格式的,手动resolve比较烦心。

core data还有其他sql所不具备的优点,比如对undo的支持,多个context实现sketchbook类似的功能。为ManagedObject优化的row cash等。

另外core data是支持多线程的,但需要thread confinement的方式实现,使用了多线程之后可以最大化的防止阻塞主线程


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/bake/11417718.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-15
下一篇 2023-05-15

发表评论

登录后才能评论

评论列表(0条)

保存