第二步:为项目添加 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的方式实现,使用了多线程之后可以最大化的防止阻塞主线程
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)