这是苹果官方文档 Core Data Programming Guide 的渣翻译。
在iOS和OS X上,一个应用的生命周期开端略有不同。
当一个OS X应用花费了很长的一段时间去启动并且反应越来越慢的时候,系统会改变光标来展示状态。用户可以决定是否等待应用启动完毕亦或是关闭这个应用。
在iOS上,却没有这个概念。如果一个应用在既定时间内没有启动完毕,操作系统就会终止这个应用。因此,尽可能快地完成一个应用的启动过程是十分关键的。
另一方面,作为应用生命周期第一步的Core Data初始化过程,只有完成了这个过程应用才能访问Core Data数据,你当然是希望越快越好。虽然不是经常发生,但是Core Data偶尔会花费比平常更多的时间去完成初始化。
因此,在iOS应用中推荐使用把启动过程分成两部分,来避免应用被终止:
- 一个尽可能小的能够告诉用户应用正在启动的过程。
- 在Core Data完成初始化后立刻完成UI加载。
在iOS中初始化Core Data
第一步就是修改application:didFinishLaunchingWithOptions:方法的实现。在application:didFinishLaunchingWithOptions:方法中需要初始化Core Data还有其他一些琐碎的事。如果你正在使用storyboard,你可以进一步在这里设置启动图片。
作为Core Data初始化的一部分,可以把配置持久化存取区(NSPersistentStore)到持久化存储协调器(NSPersistentStoreCoordinator)这一过程放到后台队列中运行。这个过程可能会耗费不知多长的时间,如果在主线程运行会阻塞用户界面,可能还会导致应用被杀死。
当持久化存储区被配置到了持久化存储协调器之后,你就可以重新回到主线程让用户界面构建完毕并且展示给用户。
把Core Data从Application Delegate分离
在之前的iOS版本中,Core Data栈都是在application delegate里面初始化的。然而,这样做会导致大量的代码混杂在应用声明周期事件处理逻辑之中。
推荐让Core Data栈在一个自己创建的一个控制器里面初始化,并且由application delegate创建并持有这个控制器的引用。这样做可以提升Core Data的内聚性并且让application delegate更加整洁。这个独立的控制器细节可以在之前1.2章节里面找到。
从“1.2 初始化Core Data栈”章节中整合代码到iOS应用中,并且在application delegate添加一个属性,在applicationDidFinishLaunching生命周期方法中创建这个控制器并引用它。
OBJECTIVE-C
@interface AppDelegate : UIResponder <UIApplicationDelegate>
@property (strong, nonatomic) UIWindow *window;
@property (strong, nonatomic) DataController *dataController;
@end
@implementation AppDelegate
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
[self setDataController:[[DataController alloc] init];
// Basic User Interface initialization
return YES;
}
@end
SWIFT
class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
var dataController: DataController!
func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
dataController = DataController()
// Basic User Interface initialization
return true
}
利用在completion block中创建、初始化一个独立的控制器,你成功将Core Data栈从application delegate分离了出来,并且还给application delegate配置了一个回调,以便用户界面接口能够知道什么时候开始请求数据。