Core Data 基礎

最近做聊天記錄緩存的東西,用到了數據庫的.本想著用FMDB因為之前用過,回來看了看Core Data也是不錯的,今天想把遇到的坑記錄一下.

  • 首先說,為什么用到數據庫?
    如果只做數據緩存,那么用歸檔或者將Json寫到文件就可以了.若用到了增刪改查,那么就要用到數據庫了.
  • 準備工作:
    1.創建一個文件


    373868DF-90CF-4065-921A-EF5A4B0425EF.png

    注意:每個工程的Data Model 文件必須新建一個,新建該文件會和該工程關聯,如果只是拖入了一個該文件會報錯.

2.建立數據表

屏幕快照 2016-04-11 下午3.35.46.png
  1. Add Entity添加實體 ->相當于Advice_Flag對象
  2. Add Attribute添加屬性 ->name...屬性
    注意:要有有key1,key2... 這個為保留字段防止以后加入字段.因為如果和本地的數據庫表不一致的話,會引起程序的崩潰.
  • 3.數據庫存的對象是NSManagedObject,所以你寫入數據庫的是對象是Student,根據你創建Entity,選中Student實體
屏幕快照 2016-03-09 上午10.45.30.png

生成對應的文件

屏幕快照 2016-04-11 下午4.22.31.png

你會發現Student是繼承于NSManagedObject的子類,現在存數據庫的Model已經準備好了

  • 那么我們開始創建數據庫代碼如下:
@interface CoreDataDAO : NSObject

  //被管理的對象上下文
  @property (readonly, strong, nonatomic) NSManagedObjectContext *managedObjectContext;
//被管理的對象模型
@property (readonly, strong, nonatomic) NSManagedObjectModel *managedObjectModel;
//持久化存儲協調者
@property (readonly, strong, nonatomic) NSPersistentStoreCoordinator *persistentStoreCoordinator;
  - (NSURL *)applicationDocumentsDirectory;
  - (void)saveContext;
  - (NSManagedObjectContext*)managedObjectContext;
  - (NSPersistentStoreCoordinator*)persistentStoreCoordinator;
  - (NSManagedObjectModel*)managedObjectModel;
  @end
#import "CoreDataDAO.h"
  @implementation CoreDataDAO
  @synthesize managedObjectContext = _managedObjectContext;
  @synthesize managedObjectModel = _managedObjectModel;
  @synthesize persistentStoreCoordinator = _persistentStoreCoordinator;
//返回 被管理的對象上下文
 - (NSManagedObjectContext *)managedObjectContext
{
    if (_managedObjectContext) {
        return _managedObjectContext;
    }
    
    NSPersistentStoreCoordinator *coordinator = [self persistentStoreCoordinator];
    if (coordinator) {
        _managedObjectContext = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSMainQueueConcurrencyType];
        [_managedObjectContext setPersistentStoreCoordinator:coordinator];
    }
    return _managedObjectContext;
}
// 返回 持久化存儲協調者
  - (NSPersistentStoreCoordinator *)persistentStoreCoordinator
{
    if (_persistentStoreCoordinator) {
        return _persistentStoreCoordinator;
    }
    
    //數據庫文件
    NSURL *storeURL = [[self applicationDocumentsDirectory] URLByAppendingPathComponent:@"NHDoctor.sqlite"];
    
    _persistentStoreCoordinator = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:[self managedObjectModel]];
    
    [_persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreType
                                              configuration:nil
                                                        URL:storeURL
                                                    options:nil
                                                      error:nil];
    
    return _persistentStoreCoordinator;
}
//  返回 被管理的對象模型
  - (NSManagedObjectModel *)managedObjectModel
{
    if (_managedObjectModel) {
        return _managedObjectModel;
    }
    //模型文件
    NSURL *modelURL = [[NSBundle mainBundle] URLForResource:@"你的文件名" withExtension:@"momd"];
    _managedObjectModel = [[NSManagedObjectModel alloc] initWithContentsOfURL:modelURL];
    return _managedObjectModel;
}
#pragma mark - 應用程序沙箱
// 返回應用程序Docment目錄的NSURL類型
  - (NSURL *)applicationDocumentsDirectory
{
    return [[[NSFileManager defaultManager] URLsForDirectory:NSDocumentDirectory inDomains:NSUserDomainMask] lastObject];
}
  - (void)saveContext {
    NSManagedObjectContext *managedObjectContext = self.managedObjectContext;
    if (managedObjectContext != nil) {
        NSError *error = nil;
        if ([managedObjectContext hasChanges] && ![managedObjectContext save:&error]) {
            // Replace this implementation with code to handle the error appropriately.
            NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
            abort();
        }
    }
}
@end

內容結構如下:


屏幕快照 2016-04-11 下午4.29.02.png
  • 注意實體的屬性名(Student的name字段)最好合后臺的字段一樣.因為這樣可以把數據庫里的字段拼成后臺返回的數據格式再次解析成你能用的Model.

具體如代碼注釋,基類已經創建好了.那么我們還需要一個管理類Core Data Manager,負責的增刪改查.

假設你的項目使用的是StudentModel,數據庫存儲的model 是Student StudentModel字段最好跟Student字段名一致,不一致需要為Student字段賦值,如圖:

屏幕快照 2016-04-11 下午4.48.16.png

你需要將你的StudentModel 轉換成 Student類的Model, Student用KVC賦值:

- (void)setValue:(nullable id)value forKey:(NSString *)key;

  • 插入數據
- (BOOL)insertData:(id)dataModel{
    //配置項(文件名)
    CoreDataConfig *config = [CoreDataConfig defaulter];
    //config.tableName 表名
    //[self managedObjectContext] 上下文,都在配置項里配置好了
    Student *student= [NSEntityDescription insertNewObjectForEntityForName:config.tableName inManagedObjectContext:[self managedObjectContext]];
    //對student插入內容賦值
    [self addDataWithModel:student withDic:dataModel];
    //保存
    return  [self saveContext];
}
用的時候導入
#import <objc/runtime.h>
- (void)addDataWithModel:(id)sourceData withDic:(id)incomingData{
    
    u_int count;
    objc_property_t *properties=class_copyPropertyList([sourceData class], &count);
    for (int i = 0; i < count ; i++)
    {
        const char* propertyName =property_getName(properties[i]);
        NSString *strName = [NSString stringWithCString:propertyName encoding:NSUTF8StringEncoding];
        if ([incomingData valueForKey:strName] != nil) {
            [sourceData setValue:[incomingData valueForKey:strName] forKey:strName];
        }
        
    }
}

這樣插入就做好了(匠心),再去沙盒去看看數據,字段改變一定要刪除沙盒的數據庫要不會崩潰.

  • 刪除 代碼如下:
-(BOOL)deleteData:(id)dataModel{
    CoreDataConfig *config = [CoreDataConfig defaulter];

    NSEntityDescription *entityDescription = [NSEntityDescription entityForName:config.tableName inManagedObjectContext:self.managedObjectContext];
    
    NSFetchRequest *request = [[NSFetchRequest alloc] init];
    [request setEntity:entityDescription];
    
    NSPredicate *predicate = [NSPredicate predicateWithFormat:
                              @"name =  %@", ((Student*)dataModel).name];
    [request setPredicate:predicate];
    
    NSError *error = nil;
    NSArray *listData = [self.managedObjectContext executeFetchRequest:request error:&error];
    BOOL is = NO;
    for (id data in listData) {
        [self.managedObjectContext deleteObject:data];
        [self saveContext];
        is = YES;
    }
    return is;
}
-(BOOL)updataData:(id)dataModel{
    CoreDataConfig *config = [CoreDataConfig defaulter];

    NSFetchRequest* fetchRequest = [[NSFetchRequest alloc] init];
    [fetchRequest setEntity:[NSEntityDescription entityForName:config.tableName inManagedObjectContext:self.managedObjectContext]];
    NSPredicate* predicate = nil;
    //查詢關鍵字,關鍵字studentId 不能用%@代替,必須寫出來
    predicate = [NSPredicate predicateWithFormat:@"studentId = %@",((Student*)dataModel).studentId];
    [fetchRequest setPredicate:predicate];
    
    NSError* error = nil;
    NSArray* arr = [self.managedObjectContext executeFetchRequest:fetchRequest error:&error];
    
    for (Student* data in arr) {
        [self addDataWithModel:data withDic:dataModel];
    }
    //保存
    return  [self saveContext];
}
-(NSArray*)query:(id)dataModel{
    
    CoreDataConfig *config = [CoreDataConfig defaulter];

    NSEntityDescription* entity = [NSEntityDescription entityForName:config.tableName inManagedObjectContext:self.managedObjectContext];
    NSFetchRequest* fetchRequest = [[NSFetchRequest alloc] init];
    [fetchRequest setEntity:entity];
    
    //組合謂詞用AND    ,studentId = %@ AND name = %@  ,'=' 可以換成< 等符號
    NSPredicate* predicate;
    predicate = [NSPredicate predicateWithFormat:@"studentId = %@",((Student*)dataModel).studentId];

    [fetchRequest setPredicate:predicate];
    //排序
    NSSortDescriptor* sort = [NSSortDescriptor sortDescriptorWithKey:@"studentId" ascending:NO];
    [fetchRequest setSortDescriptors:@[sort]];
    //查詢個數
    [fetchRequest setFetchLimit:10];
    
    
    NSError* error = nil;
    NSArray* arr = [self.managedObjectContext executeFetchRequest:fetchRequest error:&error];
    
    return arr;
    
}

到此增刪改查已經寫好,還有一個重要的一點,增刪改查你得知道什么時候發生的,所以必須得有一個數據改動的回調函數,蘋果幫我們寫好了這個函數 NSFetchedResultsController

-(NSFetchedResultsController*)creatFetchedResultWithController:(id)delegate withDic:(id)dataModel{
    CoreDataConfig *config = [CoreDataConfig defaulter];

    NSFetchRequest* request = [[NSFetchRequest alloc] init];
    NSEntityDescription* entity = [NSEntityDescription entityForName:config.tableName inManagedObjectContext:self.managedObjectContext];
    [request setEntity:entity];
    NSPredicate *predicate = [NSPredicate predicateWithFormat:
                              @"name =  %@", ((Student*)dataModel).name];    [request setPredicate:predicate];
    //排序必須寫的
    NSSortDescriptor* sort = [NSSortDescriptor sortDescriptorWithKey:@"name" ascending:YES];
    [request setSortDescriptors:@[sort]];
    NSFetchedResultsController* controller = [[NSFetchedResultsController alloc] initWithFetchRequest:request managedObjectContext:self.managedObjectContext sectionNameKeyPath:nil cacheName:nil];
    controller.delegate = delegate;
    NSError* error = nil;
    if ([controller performFetch:&error]) {
        
    }
    //返回的 controller 一定要有持有者
    return controller;
}

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.

    StudentModel *student = [[StudentModel alloc] init];
    student.name = @"小明";
    self.fetchedController = [[CoreDataManager sharedManager] creatFetchedResultWithController:self withDic:student];
}

- (void)controller:(NSFetchedResultsController *)controller
   didChangeObject:(id)anObject
       atIndexPath:(nullable NSIndexPath *)indexPath
     forChangeType:(NSFetchedResultsChangeType)type
      newIndexPath:(nullable NSIndexPath *)newIndexPath{
    
}
  • 注意一定要self.fetchedController接收函數的返回對象,不然代理函數不會走.
以上是Core Data的入門,寫不好的地方多多包涵,希望能幫助到用Core Data的同學,后續會更新Core Data 的多線程安全,謝謝大家
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,431評論 6 544
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,637評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,555評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,900評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,629評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,976評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,976評論 3 448
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,139評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,686評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,411評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,641評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,129評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,820評論 3 350
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,233評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,567評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,362評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,604評論 2 380

推薦閱讀更多精彩內容