轉 iOS 資料持久化 -- Core Data (2)
CoreData
1.Core Data 是資料持久化存儲的最佳方式
2.資料最終的存儲類型可以是:SQLite資料庫,XML,二進制,記憶體裡,或自定義資料類型
在Mac OS X 10.5Leopard及以後的版本中,開發者也可以通過繼承NSPersistentStore類以建立自定義的存儲格式
3.好處:能夠合理管理記憶體,避免使用sql的麻煩,高效
4.構成:
(1)NSManagedObjectContext(被管理的資料上下文)
操作實際内容(操作持久層)
作用:插入資料,查詢資料,删除資料
(2)NSManagedObjectModel(被管理的資料模型)
資料庫所有表格或資料結構,包含各實體的定義資訊
作用:添加實體的屬性,建立屬性之間的關系
操作方法:視圖編輯器,或代碼
(3)NSPersistentStoreCoordinator(持久化存儲助理)
相當于資料庫的連接配接器
作用:設定資料存儲的名字,位置,存儲方式,和存儲時機
(4)NSManagedObject(被管理的資料記錄)
相當于資料庫中的表格記錄
(5)NSFetchRequest(擷取資料的請求)
相當于查詢語句
(6)NSEntityDescription(實體結構)
相當于表格結構
(7)字尾為.xcdatamodeld的包
裡面是.xcdatamodel檔案,用資料模型編輯器編輯
編譯後為.momd或.mom檔案
5.依賴關系
二、基于SQLite資料庫時,Core Data的簡單使用
和SQLite的差別:隻能取出整個實體記錄,然後分解,之後才能得到實體的某個屬性
1.建構流程
包括:建立資料上下文,建立資料模型,建立資料持久化存儲助理
(1)若是建立的工程,選擇空白應用程式,next
勾選Use Core Data選項
此時生成的工程檔案AppDelegate中,會自動生成被管理的資料上下文等相關代碼
(2)比如AppDelegate.h檔案中,自動生成
@property (readonly, strong, nonatomic) NSManagedObjectContext *managedObjectContext;
@property (readonly, strong, nonatomic) NSManagedObjectModel *managedObjectModel;
@property (readonly, strong, nonatomic) NSPersistentStoreCoordinator *persistentStoreCoordinator;
- (void)saveContext;
- (NSURL *)applicationDocumentsDirectory;
方法saveContext表示:儲存資料到持久層(資料庫)
方法applicationDocumentsDirectory表示:應用程式沙箱下的Documents目錄路徑
(例如/var/mobile/Applications/5FG80A45-DFB5-4087-A1B1-41342A977E21/Documents/)
(3)比如AppDelegate.h檔案中,自動生成
@synthesize managedObjectContext = __managedObjectContext;
@synthesize managedObjectModel = __managedObjectModel;
@synthesize persistentStoreCoordinator = __persistentStoreCoordinator;
儲存資料到持久層
- (void)applicationWillTerminate:(UIApplication *)application
{
[self saveContext];
}
- (void)saveContext
{
NSError *error = nil;
NSManagedObjectContext *managedObjectContext = self.managedObjectContext;
if (managedObjectContext != nil) {
if ([managedObjectContext hasChanges] && ![managedObjectContext save:&error]) {
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
abort();
}
}
}
Documents目錄路徑
- (NSURL *)applicationDocumentsDirectory
{
return [[[NSFileManager defaultManager] URLsForDirectory:NSDocumentDirectory inDomains:NSUserDomainMask] lastObject];
}
被管理的資料上下文
初始化的後,必須設定持久化存儲助理
- (NSManagedObjectContext *)managedObjectContext
{
if (__managedObjectContext != nil) {
return __managedObjectContext;
}
NSPersistentStoreCoordinator *coordinator = [self persistentStoreCoordinator];
if (coordinator != nil) {
__managedObjectContext = [[NSManagedObjectContext alloc] init];
[__managedObjectContext setPersistentStoreCoordinator:coordinator];
}
return __managedObjectContext;
}
被管理的資料模型
初始化必須依賴.momd檔案路徑,而.momd檔案由.xcdatamodeld檔案編譯而來
- (NSManagedObjectModel *)managedObjectModel
{
if (__managedObjectModel != nil) {
return __managedObjectModel;
}
NSURL *modelURL = [[NSBundle mainBundle] URLForResource:@"TestApp" withExtension:@"momd"];
__managedObjectModel = [[NSManagedObjectModel alloc] initWithContentsOfURL:modelURL];
return __managedObjectModel;
}
持久化存儲助理
初始化必須依賴NSManagedObjectModel,之後要指定持久化存儲的資料類型,預設的是NSSQLiteStoreType,即SQLite資料庫;并指定存儲路徑為Documents目錄下,以及資料庫名稱
- (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;
}
如果不是新工程,也可以自己寫入相關代碼
(4)此外還生成了TestApp.xcdatamodeld檔案
(5)還自動連結了CoreData.framework
(6)在預編譯頭.pch檔案中,加入導入了CoreData.h頭檔案
#import <CoreData/CoreData.h>
2.建立資料模型(資料模型編輯器操作)
(1)建立實體
選中.xcodedatamodel對象
在右邊的資料模型編輯器的底部工具欄點選Add Entity添加實體
在最右側欄對實體命名
(2)建立實體屬性
選中實體,點選底部工具欄的Add Attribute添加屬性
選中新添加的屬性,對屬性進行命名,并設定屬性的資料類型Attribute Type
(3)為兩個實體建立關系
選中一個實體,在底部工具欄點選Add Relationship添加關系
選中新關系,對關系添加名稱,目标destination設定為另個實體
(4)建立傳回關系
(當你建立一個目标關系,最好建立一個傳回關系)
在另一個實體中建立一個關系并命名,設定目标對象為這之前的實體
并在Inverse屬性選這之前的關系名稱
(5)設定兩個關系的删除規則Delete Rule,都為關聯模式
關聯模式cascade:其中一個資料被删除,另一個實體中的資料也會跟着删除
(6)最終兩個對象的關系圖為
切換Editor Stype按鈕
會看到另一種編輯方式:
3.插入資料
在AppDelegate.m的application:didFinishLaunchingWithOptions:方法裡,調用自定義方法
insertCoreData插入資料,代碼如下:
- (void)insertCoreData
{
NSManagedObjectContext *context = [self managedObjectContext];
NSManagedObject *contactInfo = [NSEntityDescription insertNewObjectForEntityForName:@"ContactInfo" inManagedObjectContext:context];
[contactInfo setValue:@"name B" forKey:@"name"];
[contactInfo setValue:@"birthday B" forKey:@"birthday"];
[contactInfo setValue:@"age B" forKey:@"age"];
NSManagedObject *contactDetailInfo = [NSEntityDescription insertNewObjectForEntityForName:@"ContactDetailInfo" inManagedObjectContext:context];
[contactDetailInfo setValue:@"address B" forKey:@"address"];
[contactDetailInfo setValue:@"name B" forKey:@"name"];
[contactDetailInfo setValue:@"telephone B" forKey:@"telephone"];
[contactDetailInfo setValue:contactInfo forKey:@"info"];
[contactInfo setValue:contactDetailInfo forKey:@"details"];
NSError *error;
if(![context save:&error])
{
NSLog(@"不能儲存:%@",[error localizedDescription]);
}
}
建立資料上下文,調用insertNewObjectForName方法,建立兩個資料記錄NSManagedObject,然後就可以對之前資料模型編輯視圖中定義的屬性進行指派。此時的資料隻在記憶體中被修改,最後調用資料上下文的save方法,儲存到持久層
4.查詢資料
在調用了insertCoreData之後,可以調用自定的查詢方法dataFetchRequest來查詢插入的資料
- (void)dataFetchRequest
{
NSManagedObjectContext *context = [self managedObjectContext];
NSFetchRequest *fetchRequest = [[[NSFetchRequest alloc] init] autorelease];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"ContactInfo" inManagedObjectContext:context];
[fetchRequest setEntity:entity];
NSError *error;
NSArray *fetchedObjects = [context executeFetchRequest:fetchRequest error:&error];
for (NSManagedObject *info in fetchedObjects) {
NSLog(@"name:%@", [info valueForKey:@"name"]);
NSLog(@"age:%@", [info valueForKey:@"age"]);
NSLog(@"birthday:%@", [info valueForKey:@"birthday"]);
NSManagedObject *details = [info valueForKey:@"details"];
NSLog(@"address:%@", [details valueForKey:@"address"]);
NSLog(@"telephone:%@", [details valueForKey:@"telephone"]);
}
}
fetchRequest相當于sql查詢語句的包裝類,需要用setEntity方法,來指定具體查詢的實體結構(表結構)
通過NSEntityDescription的entityForName方法來,傳回指向該具體實體結構的指針
然後調用executeFetchRequest:error:方法,來執行查詢操作,如果操作成功,則傳回對應的資料記錄數組
其中,可以通過NSManagedObject資料記錄對象裡關聯的屬性,查詢另一個資料記錄對象裡的屬性
5.資料模版
為每個實體生成一個NSManagedObject子類
上面設定資料和擷取資料時,使用的是Key-Value方式,更好的方法是通過生成強類型的NSManagedObject的子類,通過類的成員屬性來通路和擷取資料
(1)在資料編輯器視圖中選中實體對象,
選則file菜單,點選new,點選file...,選擇Core Data項,選擇NSManagedObject subclass,生成該實體同名的類,
繼承于NSManagedObject
生成對應的ContactInfo.h檔案
#import <Foundation/Foundation.h>
#import <CoreData/CoreData.h>
@class ContactDetailInfo;
@interface ContactInfo : NSManagedObject
@property (nonatomic, retain) NSString * age;
@property (nonatomic, retain) NSString * birthday;
@property (nonatomic, retain) NSString * name;
@property (nonatomic, retain) ContactDetailInfo *details;
@end
和ContactInfo.m檔案
其中,@dynamic告訴編譯器不做處理,使編譯通過,其getter和setter方法會在運作時動态建立,由Core Data架構為此類屬性生成存取方法
#import "ContactInfo.h"
#import "ContactDetailInfo.h"
@implementation ContactInfo
@dynamic age;
@dynamic birthday;
@dynamic name;
@dynamic details;
@end
以及ContactDetailInfo.h檔案
#import <Foundation/Foundation.h>
#import <CoreData/CoreData.h>
@class ContactInfo;
@interface ContactDetailInfo : NSManagedObject
@property (nonatomic, retain) NSString * address;
@property (nonatomic, retain) NSString * name;
@property (nonatomic, retain) NSString * telephone;
@property (nonatomic, retain) ContactInfo *info;
@end
和ContactDetailInfo.m檔案
#import "ContactDetailInfo.h"
#import "ContactInfo.h"
@implementation ContactDetailInfo
@dynamic address;
@dynamic name;
@dynamic telephone;
@dynamic info;
@end
此時,資料模型編輯器視圖最右邊欄中,實體的class就變成具體的類名
之前用Key-Value的代碼就可以修改為:
#import "ContactInfo.h"
#import "ContactDetailInfo.h"
- (void)insertCoreData
{
NSManagedObjectContext *context = [self managedObjectContext];
ContactInfo *contactInfo = [NSEntityDescription insertNewObjectForEntityForName:@"ContactInfo" inManagedObjectContext:context];
contactInfo.name = @"name B";
contactInfo.birthday = @"birthday B";
contactInfo.age = @"age B";
ContactDetailInfo *contactDetailInfo = [NSEntityDescription insertNewObjectForEntityForName:@"ContactDetailInfo" inManagedObjectContext:context];
contactDetailInfo.address = @"address B";
contactDetailInfo.name = @"name B";
contactDetailInfo.telephone = @"telephone B";
contactDetailInfo.info = contactInfo;
contactInfo.details = contactDetailInfo;
NSError *error;
if(![context save:&error])
{
NSLog(@"不能儲存:%@",[error localizedDescription]);
}
}
- (void)dataFetchRequest
{
NSManagedObjectContext *context = [self managedObjectContext];
NSFetchRequest *fetchRequest = [[[NSFetchRequest alloc] init] autorelease];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"ContactInfo" inManagedObjectContext:context];
[fetchRequest setEntity:entity];
NSError *error;
NSArray *fetchedObjects = [context executeFetchRequest:fetchRequest error:&error];
for (ContactInfo *info in fetchedObjects) {
NSLog(@"name:%@", info.name);
NSLog(@"age:%@", info.age);
NSLog(@"birthday:%@", info.birthday);
ContactDetailInfo *details = info.details;
NSLog(@"address:%@", details.address);
NSLog(@"telephone:%@", details.telephone);
}
}
三、資料庫相關
1.列印隐藏的sql語句:
在Edit Scheme中選擇Run,之後進入Arguments标簽,添加參數:“-com.apple.CoreData.SQLDebug 1”
2.使用SQLite存儲時,資料庫結構
存儲的SQLite資料庫表名稱:大寫“Z”加上實體名稱大寫,一個實體相當于一張表
具體的字段名稱:大寫“Z”加上實體屬性名稱大寫