1.2 初始化Core Data棧(Core Data Programming Guide翻譯)
這是蘋果官方文檔 Core Data Programming Guide 的渣翻譯银伟。
Core Data棧是一個框架對象集合,這個集用以作為Core Data初始化的一部分和應用中的對象和外部數(shù)據(jù)的中間層彭谁。Core Data棧會處理所有外部存儲數(shù)據(jù)的交互奉狈,因此你可以集中精力在業(yè)務邏輯上芽隆。這個棧由3部分構(gòu)成:托管對象上下文(NSManagedObjectContext抡谐,以下稱MOC)铁追,持久化存儲協(xié)調(diào)器(NSPersistentStoreCoordinator姻成,以下稱PSC)也祠,托管對象模型(NSManagedObjectModel昙楚,以下稱MOM),稍后在本章節(jié)會詳細討論诈嘿。
在訪問應用數(shù)據(jù)之前你要初始化Core Data棧堪旧。棧的初始化準備好了Core Data數(shù)據(jù)請求和數(shù)據(jù)的創(chuàng)建。這里有一個Core Data棧的創(chuàng)建例子奖亚。
代碼3-1 Core Data棧的創(chuàng)建例子
OBJECTIVE-C
@interface MyDataController : NSObject
@property (strong) NSManagedObjectContext *managedObjectContext;
- (void)initializeCoreData;
@end
@implementation MyDataController
- (id)init
{
self = [super init];
if (!self) return nil;
[self initializeCoreData];
return self;
}
- (void)initializeCoreData
{
NSURL *modelURL = [[NSBundle mainBundle] URLForResource:@"DataModel" withExtension:@"momd"];
NSManagedObjectModel *mom = [[NSManagedObjectModel alloc] initWithContentsOfURL:modelURL];
NSAssert(mom != nil, @"Error initializing Managed Object Model");
NSPersistentStoreCoordinator *psc = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:mom];
NSManagedObjectContext *moc = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSMainQueueConcurrencyType];
[moc setPersistentStoreCoordinator:psc];
[self setManagedObjectContext:moc];
NSFileManager *fileManager = [NSFileManager defaultManager];
NSURL *documentsURL = [[fileManager URLsForDirectory:NSDocumentDirectory inDomains:NSUserDomainMask] lastObject];
NSURL *storeURL = [documentsURL URLByAppendingPathComponent:@"DataModel.sqlite"];
dispatch_async(dispatch_get_global_queue( DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^(void) {
NSError *error = nil;
NSPersistentStoreCoordinator *psc = [[self managedObjectContext] persistentStoreCoordinator];
NSPersistentStore *store = [psc addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:storeURL options:options error:&error];
NSAssert(store != nil, @"Error initializing PSC: %@\n%@", [error localizedDescription], [error userInfo]);
});
}
SWIFT
import UIKit
import CoreData
class DataController: NSObject {
var managedObjectContext: NSManagedObjectContext
init() {
// This resource is the same name as your xcdatamodeld contained in your project.
guard let modelURL = NSBundle.mainBundle().URLForResource("DataModel", withExtension:"momd") else {
fatalError("Error loading model from bundle")
}
// The managed object model for the application. It is a fatal error for the application not to be able to find and load its model.
guard let mom = NSManagedObjectModel(contentsOfURL: modelURL) else {
fatalError("Error initializing mom from: \(modelURL)")
}
let psc = NSPersistentStoreCoordinator(managedObjectModel: mom)
managedObjectContext = NSManagedObjectContext(concurrencyType: .MainQueueConcurrencyType)
managedObjectContext.persistentStoreCoordinator = psc
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0)) {
let urls = NSFileManager.defaultManager().URLsForDirectory(.DocumentDirectory, inDomains: .UserDomainMask)
let docURL = urls[urls.endIndex-1]
/* The directory the application uses to store the Core Data store file.
This code uses a file named "DataModel.sqlite" in the application's documents directory.
*/
let storeURL = docURL.URLByAppendingPathComponent("DataModel.sqlite")
do {
try psc.addPersistentStoreWithType(NSSQLiteStoreType, configuration: nil, URL: storeURL, options: nil)
} catch {
fatalError("Error migrating store: \(error)")
}
}
}
}
這個例子創(chuàng)建了一個控制器對象淳梦,表示應用的持久化層。這個控制器調(diào)用了默認的“init”進行初始化昔字。作為“init”調(diào)用的一部分爆袍,“initializeCoreData”方法會被調(diào)用首繁,之后它進行了Core Data棧的創(chuàng)建。
NSManagedObjectModel
NSManagedObjectModel實例描述了Core Data棧將要訪問的數(shù)據(jù)陨囊。在Core Data棧的創(chuàng)建中弦疮,NSManagedObjectModel(MOM)作為Core Data棧創(chuàng)建的第一步,被加載到了內(nèi)存中蜘醋。上面的源碼中使用了一個已知的NSManagedObjectModel文件名(這里是DataModel.momd)解析了一個從應用bundle中的NSURL胁塞。當這個NSManagedObjectModel被初始化,NSPersistentStoreCoordinator對象會被構(gòu)建出來压语。
NSPersistentStoreCoordinator
NSPersistentStoreCoordinator位于Core Data棧的中間啸罢。這個協(xié)調(diào)器負責實例化在模型中定義的Entity。它會創(chuàng)建了模型中的Entity胎食,和在持久化存儲(NSPersistentStore)中查詢已經(jīng)存在的實例扰才。持久化存儲可以是硬盤或者內(nèi)存〔蘖基于應用的結(jié)構(gòu)训桶,雖然不常見,但是可以允許存在多個持久化存儲被NSPersistentStoreCoordinator管理酣倾。
基于NSManagedObjectModel定義了數(shù)據(jù)結(jié)構(gòu),NSPersistentStoreCoordinator會從持久化數(shù)據(jù)中實例化對象谤专,并向下傳遞給發(fā)出請求的NSManagedObjectContext躁锡。NSPersistentStoreCoordinator還會驗證數(shù)據(jù)和NSManagedObjectModel中的定義的一致性。
NSManagedObjectContext
托管對象上下文(NSManagedObjectContext)是應用中交互最多的對象置侍,因此是暴露給整個應用的映之。可以想象MOC為一個智能的草稿板蜡坊。當你在持久化存儲中抓取了一個對象杠输,你就提取了一個臨時拷貝到了組織了一個對象圖(或者對象圖的集合)的草稿板中。你可以任意修改這個對象秕衙。直到執(zhí)行了save操作蠢甲,否則持久化存儲被會被修改。
所有托管對象(MO)必須在MOC中注冊据忘。你可以使用MOC在對象圖中添加或者刪除對象鹦牛。這個MOC會跟蹤任何你做的修改,無論是個別的對象字段還是對象之間的關系勇吊。利用修改追蹤曼追,MOC提供了撤銷(undo)和重做(redo)功能。它也保證了你修改了對象之間的關系汉规,對象圖也能同步礼殊。
如果你選擇對修改進行save操作,MOC會驗證你的對象是在合法狀態(tài)。如果是晶伦,修改會被保存到持久化存儲中碟狞,新創(chuàng)建的對象會被作為新紀錄插入,刪除對象會刪除對應的記錄坝辫。
如果沒有Core Data篷就,你需要手動寫方法來實現(xiàn)數(shù)據(jù)的歸檔和讀檔、保持跟蹤模型數(shù)據(jù)近忙、和撤銷管理器交互進行撤銷操作竭业。在Core Data框架中,主要通過MOC及舍,大部分這些功能都會自動提供未辆。