接上篇
关于Core Data的使用我在开篇的时候就介绍过他的优点,其中很重要的一点就是通过schema migration工具可以简化应对数据库结构变化的任务,并且可以高效率的数据库原地迁移工作。本篇内容我们介绍如何实现Core Data的数据迁移能力。
假定一个场景,随着业务发展的需求,我们需要调整我们的数据存储结构,还是以之前文章中的例子TestCoreData来做演示。具体步骤如下:
1、 新建Model文件(BookModel 2.xcdatamodel)
a) 新增Model版本
b) 输入Model名称,点击“Finish”
c) 会在工程目录中生成如下所示文件,有绿色对勾图标的是当前正在使用的Model文件
d) 修改新增的Model的结构(BookModel 2.xcdatamodeld)
1) AuthorEntity中,新增Attribute“married”
2) BookEntity中,删除Attribute“author”
e) 将新增的Model设置为当前使用的Model,如下图
设置完成后,可以看到绿色对勾图标已经标示在BookModel 2.xcdatamodel文件上了
按照前面文章《iPhone应用程序开发使用Core Data(一)》中介绍的方式,重新生成对应表的Entity类。这里需要指出的是,由于对Entity做的修改,比如增删Attribute后,需要相应调整使用类(如BookManager)中对应的操作代码,否则可能造成编译不过。
做完这样的调整之后,我们编译代码运行应用程序发现奔溃退出,可以看到提示错误:reason = "The model used to open the store isincompatible with the one used to create the store"
这是什么原因呢?看下边关于数据迁移的介绍。
2、 轻量级迁移数据(Lightweight Migration)
轻量迁移的条件:
a) 增加一个字段
b) 删除一个字段
c) 把一个必填字段改为可选字段
d) 把可选字段改为必填字段(但一定要定义默认值)
e) 重命名一个entity或者property
Core Data支持我们在少量改动数据结构的情况下,做很少代码的改动,将历史数据迁移到新数据Model中。例如我们上边所做的改动,就适用于轻量级数据迁移。
代码中需要做如下修改:
- (NSPersistentStoreCoordinator *)persistentStoreCoordinator
{
if(_persistentStoreCoordinator != nil) {
return_persistentStoreCoordinator;
}
_persistentStoreCoordinator= [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel: [selfmanagedObjectModel]];
NSURL *storeURL = [[selfapplicationDocumentsDirectory]URLByAppendingPathComponent:@"BookModel.sqlite"];
// handle db upgrade
NSDictionary *options = [NSDictionary dictionaryWithObjectsAndKeys:
[NSNumbernumberWithBool:YES], NSMigratePersistentStoresAutomaticallyOption,
[NSNumbernumberWithBool:YES], NSInferMappingModelAutomaticallyOption, nil];
NSError *error;
if(![_persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreTypeconfiguration:nil URL:storeURL options:options error:&error]) {
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
abort();
}
return_persistentStoreCoordinator;
}
按照上面所述修改代码后,我们再编译运行应用程序,会发现可成功启动程序,并且数据库中的历史数据也被迁移过来了。
对于结构复杂的数据库,当我们做了除上边所述的结构调整后,不适用Lightweight Migration方法了,该怎么办呢?Core Data给我们提供了更为强大的功能,我们可以定制数据迁移的过程(Customizing the Migration Process),这就是我们下篇文章要给大家介绍的内容,敬请期待!
示例代码:TestCoreData
参考文献《Xcode 4.6 doc set》