CoreData 是 iOS3.0 時引入的一個數(shù)據(jù)持久化的框架。他與 sqlite 對比最大的優(yōu)點莫過于支持對象的存儲了,蘋果的官方文檔說其簡化了數(shù)據(jù)庫的操作(個人用起來感覺依然沒有 FMDB 簡單)谈火,使用 CoreData 確實可以大量減少代碼中的 SQL 語句。
先來介紹一下 CoreData 里面一些基礎(chǔ)性的東西吧:
Core Data 基本結(jié)構(gòu)
NSPersistentStoreCoordinator
NSPersistentStoreCoordinator是一個位于本地存儲文件與緩存層(NSManagedObjectContext)之間的一個持久化層界拦,他是真實操作數(shù)據(jù)庫本地文件乍楚。
其初始化主要為兩個步驟:
- 用已經(jīng)初始化好的NSManagedObjectModel去初始化它
- 選擇NSPersistentStore為存儲的方式
- (NSPersistentStoreCoordinator *)persistentStoreCoordinator{
if (__persistentStoreCoordinator != nil) {
return __persistentStoreCoordinator;
}
NSURL *storeURL = [[self applicationDocumentsDirectory] URLByAppendingPathComponent:@"TestApp.sqlite"];
NSError *error = nil;
__persistentStoreCoordinator = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:[self managedObjectModel]];
if (![__persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:storeURL options:nil error:&error]) {
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
abort();
}
return __persistentStoreCoordinator;
}
NSPersistentStoreCoordinator的數(shù)據(jù)存儲形式目前支持三種:SQLite、二進制丛楚、XML
NSManagedObjectContext
NSManagedObjectContext是一個被管理數(shù)據(jù)的上下文族壳,他實際上是對你所有數(shù)據(jù)庫操作的一個緩存層,把你所有的操作都先緩存起來避免大量磁盤 IO 造成不流暢趣些,你在操作完數(shù)據(jù)庫后調(diào)用其save方法仿荆,就可以把你的數(shù)據(jù)庫操作提交給持久化層(NSPersistentStoreCoordinator),由持久化層一次性寫入數(shù)據(jù)庫文件坏平。初始化:
- (NSManagedObjectContext *)managedObjectContext{
if (__managedObjectContext != nil) {
return __managedObjectContext;
}
NSPersistentStoreCoordinator *coordinator = [self persistentStoreCoordinator];
if (coordinator != nil) {
__managedObjectContext = [[NSManagedObjectContext alloc] init];
[__managedObjectContext setPersistentStoreCoordinator:coordinator];
}
return __managedObjectContext;
}
NSManagedObjectModel
NSManagedObjectModel可以把它看做為一個 Core Data 的 schema拢操。生成這個類的來源就是在 xCode 里的.xcdatamodeld文件,我們可以可視化的對這個文件進行操作舶替,實際上這個文件也就相當(dāng)于數(shù)據(jù)庫的 schema令境,這個文件編譯后就是.momd或.mom文件。我們可以根據(jù)這個初始化NSManagedObjectModel:
- (NSManagedObjectModel *)managedObjectModel{
if (__managedObjectModel != nil) {
return __managedObjectModel;
}
NSURL *modelURL = [[NSBundle mainBundle] URLForResource:@"TestApp" withExtension:@"momd"];
__managedObjectModel = [[NSManagedObjectModel alloc] initWithContentsOfURL:modelURL];
return __managedObjectModel;
}
NSManagedObject
NSManagedObject是被管理的數(shù)據(jù)記錄顾瞪。對應(yīng)著就是數(shù)據(jù)庫的一個表舔庶,在我們適用 xCode 編輯完數(shù)據(jù)庫的表結(jié)構(gòu)之后可以通過xCode->Editor->Create NSManagedObject Subclass...來創(chuàng)建對應(yīng)表的 model
NSFetchRequest
NSFetchRequest就相當(dāng)于是 SQL 里面的查詢語句抛蚁,我們使用它替代了 sql 語句。這個支持謂詞查詢
NSEntityDescription
NSEntityDescription相當(dāng)于表結(jié)構(gòu)惕橙。里面包含Attributes瞧甩、Relationships、Fetched Properties三部分內(nèi)容弥鹦。也就是你在編輯.ecdatamodeld右邊的三組內(nèi)容
參考
iphone數(shù)據(jù)存儲之-- Core Data的使用(一)
Core Data淺談系列之一:基礎(chǔ)結(jié)構(gòu)
Core Data 多線程
簡介
在應(yīng)用中設(shè)計數(shù)據(jù)量大的時候必然要通過子線程來進行數(shù)據(jù)庫相關(guān)操作肚逸,這就涉及到線程安全問題了。其多線程下的使用這里有三中方案:
須知道彬坏,CoreData的NSPersistentStoreCoordinator和NSManagedObjectContext對象都是不能跨線程使用的吼虎,我們注意到:
- (id)initWithConcurrencyType:(NSManagedObjectContextConcurrencyType)ct
NSManagedObjectContextConcurrencyType 一共有三種:
- NSConfinementConcurrencyType
- NSPrivateQueueConcurrencyType
- NSMainQueueConcurrencyType
NSPrivateQueueConcurrencyType與NSMainQueueConcurrencyType分別對應(yīng)綁定一個后臺線程與主線程〔韵剩可以使用performBlock:開執(zhí)行原來需要使用dispatch_async封裝的代碼.
NSConfinementConcurrencyType不能執(zhí)行performBlock:
NSManagedObjectContext是可以基于其他的NSManagedObjectContext的,通過setParentContext方法玷犹,可以設(shè)置另外一個NSManagedObjectContext為自己的父級混滔,這個時候子級可以訪問父級下所有的對象,而且子級NSManagedObjectContext的內(nèi)容變化后歹颓,如果執(zhí)行 save 方法坯屿,會自動的 merge 到父級NSManagedObjectContext中,也就是子級 save 后巍扛,變動會同步到父級NSManagedObjectContext领跛。當(dāng)然這個時候父級也必須再 save 一次,如果父級沒有父級了撤奸,那么就會直接向NSPersistentStoreCoordinator中寫入吠昭,如果有就會接著向再上一層的父級冒泡...
通過三個級別的NSManagedObjectContext, 一個負責(zé)在 background 更新NSPersistentStoreCoordinator胧瓜。一個用在主線程矢棚,主要執(zhí)行插入,修改和刪除操作府喳,一些小的查詢也可以在這里同步執(zhí)行蒲肋,如果有大的查詢,就起一個新的NSPrivateQueueConcurrencyType類型的NSManagedObjectContext钝满,然后放在后臺去執(zhí)行查詢兜粘,查詢完成后將結(jié)果返回主線程。
實際上這種多線程的訪問也類似 FMDB 中FMDatabaseQueue弯蚜,就是把數(shù)據(jù)庫操作全部放到一個子線程中孔轴,然后所有的操作都最終調(diào)用這個子線程來執(zhí)行
參考:
通則不痛,遠離阻塞-在多線程環(huán)境中使用CoreData碎捺,以及一個簡單的封裝
CoreData 多線程下NSManagedObjectContext的使用
代碼參考
Core Data是如何保存的
- 當(dāng)然是NSManagedObjectContext調(diào)用save方法的時候距糖。
- 這時候context里面持有的那些NSManagedObject將會自己調(diào)用自己的willSave方法玄窝。
- NSManagedObjectContextWillSaveNotification發(fā)出。
- 開始驗證悍引。這個驗證可能是你在data model里面寫的恩脂。也可以是在NSManagedObject里面代碼寫的。
- 驗證結(jié)束以后數(shù)據(jù)就合并到NSPersistentStoreCoordinator和NSPersistentStore里面趣斤。
- 如果你在合并的時候更改了數(shù)據(jù)俩块。這時候會重新驗證數(shù)據(jù)。
- 這個時候已經(jīng)把需要驗證過的合并數(shù)據(jù)存到持久化介質(zhì)當(dāng)中浓领。
- 最后NSManagedObjectContextDidSaveNotification這個通知發(fā)出玉凯。