iphone數據存儲之 Core Data的使用

一、概念

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”加上實體屬性名稱大寫

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章