數(shù)據(jù)庫多線程問題

1.FMDB線程安全的實現(xiàn)

應(yīng)用中不可在多個線程中共同使用一個FMDatabase對象操作數(shù)據(jù)庫揪垄,這樣會引起數(shù)據(jù)庫數(shù)據(jù)混亂焦辅。如果要實現(xiàn)多線程轻局,就需要使用FMDatabaseQueue。

FMDatabaseQueue *queue = [[FMDatabaseQueue alloc] initWithPath:[self getDatabasePath]];

用FMDatabaseQueue來創(chuàng)建隊列成翩,所有的操作都在block中執(zhí)行氮惯。

//插入數(shù)據(jù)

? ? [queue inDatabase:^(FMDatabase *database) {

? ? ? ? //寫入數(shù)據(jù)

? ? ? ? sql = @"insert into Test (name,image) values (?,?)";

? ? ? ? [database executeUpdate:sql,@"張三",data];

? ? }];

實現(xiàn)原理:在應(yīng)用中只有一個FMDatabaseQueue實例叮雳,所有的任務(wù)都使用這一個共同的實例想暗。FMDatabaseQueue其實是創(chuàng)建了一個gcd串行隊列,放入隊列中的block都串行執(zhí)行帘不,這就保證了同一時間隊列中只執(zhí)行一個操作说莫,避免數(shù)據(jù)的混亂。

2.事務(wù)的概念

如果要保證多個操作同時成功或同時失敗寞焙,可以把多個操作放到事務(wù)里储狭。

[queue inTransaction:^(FMDatabase *db,BOOL*rollback) {

?[db executeUpdate:@"INSERT INTO myTable VALUES (?)", [NSNumbernumberWithInt:1]];

?[db executeUpdate:@"INSERT INTO myTable VALUES (?)", [NSNumbernumberWithInt:2]];?

?[db executeUpdate:@"INSERT INTO myTable VALUES (?)", [NSNumbernumberWithInt:3]];

if(whoopsSomethingWrongHappened) { *rollback =YES;return;如果發(fā)生錯誤就進行回滾,把之前已經(jīng)執(zhí)行的任務(wù)都取消}// ...

[db executeUpdate:@"INSERT INTO myTable VALUES (?)", [NSNumbernumberWithInt:4]];}];

3.coredata多線程

1.初始化moc(托管對象上下文捣郊,NSManagedObjectContext)的時候可以指定并發(fā)隊列辽狈。

NSPrivateQueueConcurrencyType 私有并發(fā)隊列類型,操作在子線程完成呛牲。

NSMainQueueConcurrencyType 主并發(fā)隊列類型稻艰,涉及到UI相關(guān)操作應(yīng)該使用這個隊列。

2.調(diào)用方式侈净,不能直接在不同的線程里使用同一個moc,會造成數(shù)據(jù)混亂:

- (void)performBlock:(void(^)())block 異步執(zhí)行的block僧凤,調(diào)用之后會立刻返回畜侦。

- (void)performBlockAndWait:(void(^)())block 同步執(zhí)行的block,調(diào)用之后會等待這個任務(wù)完成躯保,才會繼續(xù)向下執(zhí)行旋膳。

比如說,在多線程的環(huán)境下執(zhí)行MOC的save方法途事,就是將save方法放在MOC的block體中異步執(zhí)行验懊,其他方法的調(diào)用也是一樣的。

[context performBlock:^{

? ? [context save:nil];異步執(zhí)行save

}];??:這種方法和我們之前做線程切換的方式是不一樣的尸变,不能自己開一個子線程然后到子線程里面去執(zhí)行moc义图,應(yīng)該是調(diào)用moc自身綁定的隊列,把操作放到block里去執(zhí)行召烂。

iOS5之前使用多個moc:可能有多個MOC關(guān)聯(lián)在同一個PSC上碱工,當(dāng)一個MOC發(fā)生改變并持久化到本地時,系統(tǒng)并不會將其他MOC緩存在內(nèi)存中的NSManagedObject對象改變奏夫。所以這就需要我們在MOC發(fā)生改變時怕篷,將其他MOC數(shù)據(jù)更新。

多個moc關(guān)聯(lián)同一個psc酗昼,在通知中心注冊NSNotification廊谓。

// 創(chuàng)建主隊列MOC,用于執(zhí)行UI操作NSManagedObjectContext*mainMOC = [[NSManagedObjectContextalloc] initWithConcurrencyType:NSMainQueueConcurrencyType]; mainMOC.persistentStoreCoordinator = PSC;

// 創(chuàng)建私有隊列MOC麻削,用于執(zhí)行其他耗時操作NSManagedObjectContext*backgroundMOC = [[NSManagedObjectContextalloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType]; backgroundMOC.persistentStoreCoordinator = PSC;

// 通過監(jiān)聽NSManagedObjectContextDidSaveNotification通知蒸痹,來獲取所有MOC的改變消息 [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(contextChanged:) name:NSManagedObjectContextDidSaveNotification object:nil]; ?

// MOC改變后的通知回調(diào)

- (void)contextChanged:(NSNotification *)noti {

NSManagedObjectContext *MOC = noti.object;

[MOC performBlock:^{ // 直接調(diào)用系統(tǒng)提供的同步API春弥,系統(tǒng)內(nèi)部會完成同步的實現(xiàn)細節(jié)。 [MOC mergeChangesFromContextDidSaveNotification:noti]; }];}

iOS5之前的數(shù)據(jù)同步:

簡單數(shù)據(jù)沖突电抚,一個moc數(shù)據(jù)發(fā)生改變惕稻,提交存儲區(qū),其他moc并沒有對這個改變的數(shù)據(jù)進行更新蝙叛。

有三種通知方式:

NSManagedObjectContextWillSaveNotification

NSManagedObjectContextDidSaveNotification

NSManagedObjectContextObjectsDidChangeNotification

監(jiān)聽到通知以后調(diào)用同步api俺祠,mergeChangesFromContextDidSaveNotification:noti

復(fù)雜數(shù)據(jù)同步:一個moc對本地數(shù)據(jù)的存儲做出了改變,另一個moc也對相同的數(shù)據(jù)存儲做了改變借帘,這樣在save的時候就產(chǎn)生了數(shù)據(jù)沖突蜘渣。

這種情況下可以設(shè)置moc的mergePolicy屬性來指定解決沖突的具體方案。一共有五種屬性值肺然。

NSErrorMergePolicy: 默認值蔫缸,當(dāng)出現(xiàn)合并沖突時,返回一個NSError對象來描述錯誤际起,而MOC和持久化存儲區(qū)不發(fā)生改變拾碌。

NSMergeByPropertyStoreTrumpMergePolicy:? 以本地存儲為準(zhǔn),使用本地存儲來覆蓋沖突部分街望。

NSMergeByPropertyObjectTrumpMergePolicy:? 以MOC的為準(zhǔn)校翔,使用MOC來覆蓋本地存儲的沖突部分。

NSOverwriteMergePolicy:? 以MOC為準(zhǔn)灾前,用MOC的所有NSManagedObject對象覆蓋本地存儲的對應(yīng)對象防症。

NSRollbackMergePolicy:? ? 以本地存儲為準(zhǔn),MOC所有的NSManagedObject對象被本地存儲的對應(yīng)對象所覆蓋哎甲。

iOS5之后使用多個moc:

關(guān)聯(lián)了PSC的MOC是parentMOC蔫敲,parentMOC執(zhí)行save操作時真正將數(shù)據(jù)寫入了本地數(shù)據(jù)庫。每一個parentMOC都可以有多個childMOC炭玫,但是childMOC并不會關(guān)聯(lián)parentMOC奈嘿,childMOC執(zhí)行save操作的時候并不會真正把數(shù)據(jù)寫入數(shù)據(jù)庫,而是會通知他的parentMOC自己發(fā)生了改變吞加,由parentMOC完成寫入數(shù)據(jù)庫的任務(wù)指么。

// 創(chuàng)建PSC實例對象,還是用上面Demo的實例化代碼

? ? NSPersistentStoreCoordinator *PSC = self.persistentStoreCoordinator;

// 創(chuàng)建主隊列MOC榴鼎,用于執(zhí)行UI操作 NSManagedObjectContext *mainMOC = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSMainQueueConcurrencyType]; mainMOC.persistentStoreCoordinator = PSC; ?

// 創(chuàng)建私有隊列MOC伯诬,用于執(zhí)行其他耗時操作,backgroundMOC并不需要設(shè)置PSC NSManagedObjectContext *backgroundMOC = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType]; backgroundMOC.parentContext = mainMOC;?

[backgroundMOC performBlock:^{

[backgroundMOC save:nil];

[mainMOC performBlock:^{

[mainMOC save:nil];//私有隊列調(diào)用了save方法之后巫财,主隊列也要調(diào)用save方法盗似,這樣才能實現(xiàn)本地數(shù)據(jù)持久化。

}]; }]; } ?

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末平项,一起剝皮案震驚了整個濱河市赫舒,隨后出現(xiàn)的幾起案子悍及,更是在濱河造成了極大的恐慌,老刑警劉巖接癌,帶你破解...
    沈念sama閱讀 212,185評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件心赶,死亡現(xiàn)場離奇詭異,居然都是意外死亡缺猛,警方通過查閱死者的電腦和手機缨叫,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,445評論 3 385
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來荔燎,“玉大人耻姥,你說我怎么就攤上這事∮凶桑” “怎么了琐簇?”我有些...
    開封第一講書人閱讀 157,684評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長座享。 經(jīng)常有香客問我婉商,道長,這世上最難降的妖魔是什么渣叛? 我笑而不...
    開封第一講書人閱讀 56,564評論 1 284
  • 正文 為了忘掉前任据某,我火速辦了婚禮,結(jié)果婚禮上诗箍,老公的妹妹穿的比我還像新娘。我一直安慰自己挽唉,他們只是感情好滤祖,可當(dāng)我...
    茶點故事閱讀 65,681評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著瓶籽,像睡著了一般匠童。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上塑顺,一...
    開封第一講書人閱讀 49,874評論 1 290
  • 那天汤求,我揣著相機與錄音,去河邊找鬼严拒。 笑死扬绪,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的裤唠。 我是一名探鬼主播挤牛,決...
    沈念sama閱讀 39,025評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼种蘸!你這毒婦竟也來了墓赴?” 一聲冷哼從身側(cè)響起竞膳,我...
    開封第一講書人閱讀 37,761評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎诫硕,沒想到半個月后坦辟,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,217評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡章办,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,545評論 2 327
  • 正文 我和宋清朗相戀三年锉走,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片纲菌。...
    茶點故事閱讀 38,694評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡挠日,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出翰舌,到底是詐尸還是另有隱情嚣潜,我是刑警寧澤,帶...
    沈念sama閱讀 34,351評論 4 332
  • 正文 年R本政府宣布椅贱,位于F島的核電站懂算,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏庇麦。R本人自食惡果不足惜计技,卻給世界環(huán)境...
    茶點故事閱讀 39,988評論 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望山橄。 院中可真熱鬧垮媒,春花似錦、人聲如沸航棱。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,778評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽饮醇。三九已至它抱,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間朴艰,已是汗流浹背观蓄。 一陣腳步聲響...
    開封第一講書人閱讀 32,007評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留祠墅,地道東北人侮穿。 一個月前我還...
    沈念sama閱讀 46,427評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像毁嗦,于是被迫代替她去往敵國和親撮珠。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,580評論 2 349

推薦閱讀更多精彩內(nèi)容