FMDB常用API簡述及多線程單例類封裝

FMDB的簡單使用

  • 導入頭文件 #import "FMDB.h"
  • 全局的靜態數據庫對象
    initialize
    // 這個類在第一次使用時就被調用一次,后面就不會再調用;而且是先于實例化方法調用的.
static FMDatabase *_db;
+ (void)initialize {
 // NSLog(@"這個類在第一次使用時就被調用一次"); 
// 1.獲取數據庫路徑
 NSString *SQLPath = [[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:@"heros.db"];//heros是你的表名
 // 2.創建數據庫和建表,只會執行一次 
_db = [FMDatabase databaseWithPath:SQLPath];
 // 3.打開數據庫
 BOOL isOpen = [_db open]; 
if (isOpen) { 
  // 4.建表 
  BOOL isCreate = [_db executeUpdate:@"create table if not exists t_heros(id integer primary key,name text not null,age integer)"];
   if (isCreate) {
     NSLog(@"建表成功");
    } 
  } 
}

新增

[_db executeUpdateWithFormat:@"insert into t_heros(name,age) values('張三',18)"];

修改

[_db executeUpdateWithFormat:@"update t_heros set age = 19 where name = '張三'"];

刪除

[_db executeUpdateWithFormat:@"delete from t_heros where name = '張三'"];

查詢 -- 模糊查詢

// 執行查詢語句.獲取到結果集
FMResultSet *resultSet = [_db executeQuery:@"select * from t_heros"];
// 遍歷結果集,取數據
while ([resultSet next]) {
 // 取出來的數據 
NSString *name = [resultSet stringForColumn:@"name"];
 int age = [resultSet intForColumn:@"age"];
}

如果數據庫里面的記錄有很多條,那么就要像解析模型一樣創建數據源去接收多條數據存到數據源

+ (NSArray *)selectHeros{  
  // 定義臨時的模型數據  
  NSMutableArray *tmpM = [NSMutableArray array];  
  // 執行查詢語句.獲取到結果集  
  FMResultSet *resultSet = [_db executeQuery:@"select * from t_heros"];  
  // 遍歷結果集,取數據  
  while ([resultSet next]) {       
 // 創建模型       
 Heros *hero = [[Heros alloc] init];     //自己的模型 
  // 取出來的數據    
    NSString *name = [resultSet stringForColumn:@"name"];     
   int age = [resultSet intForColumn:@"age"];  
      // 給模型賦值    ,name和age為模型里面的key,根據自己的實際情況更改
    hero.name = name;        
    hero.age = @(age);     
   // 將模型添加到模型數組      
  [tmpM addObject:hero];  
  }   
 return tmpM.copy;

到此簡單的數據庫增刪改查都可以實現,那么問題來了

dispatch_async(dispatch_get_global_queue(0, 0), ^{
 [_db executeUpdateWithFormat:@"insert into t_heros(name,age) values('李四',19)"];
});
dispatch_async(dispatch_get_global_queue(0, 0), ^{ 
[_db executeUpdateWithFormat:@"insert into t_heros(name,age) values('周五',20)"];
});

項目中不可能只是單線程的情況,那么多線程情況下就有可能出現多條數據插入,只有一條數據成功的情況,因為FMDB默認不支持多線程去操作數據庫,

分析FMDatabaseQueue
源碼

- (void)inDatabase:(void (^)(FMDatabase *db))block { 
/* Get the currently executing queue (which should probably be nil, but in theory could be another DB queue * and then check it against self to make sure we're not about to deadlock. */ 
FMDatabaseQueue *currentSyncQueue = (__bridge id)dispatch_get_specific(kDispatchQueueSpecificKey); 
assert(currentSyncQueue != self && "inDatabase: was called reentrantly on the same queue, which would lead to a deadlock");
 FMDBRetain(self); 
// 串行隊列+同步任務
 dispatch_sync(_queue, ^() {
 // 數據庫實例
 FMDatabase *db = [self database];
 block(db); 
if ([db hasOpenResultSets]) {
 NSLog(@"Warning: there is at least one open result set around after performing [FMDatabaseQueue inDatabase:]");
#if defined(DEBUG) && DEBUG 
NSSet *openSetCopy = FMDBReturnAutoreleased(
[[db valueForKey:@"_openResultSets"] copy]);
 for (NSValue *rsInWrappedInATastyValueMeal in openSetCopy) {
 FMResultSet *rs = (FMResultSet *)[rsInWrappedInATastyValueMeal pointerValue]; 
NSLog(@"query: '%@'", [rs query]);
 }
#endif 
} 
});
 FMDBRelease(self);}

結論 :
FMDatabaseQueue管理著一個隊里,這個隊列是串行隊列.
串行隊列里面裝的都是同步任務
同步任務里面就是操作數據庫的代碼
所以如果我們的 app 需要多線程操作數據庫,那么就需要使用 FMDatabaseQueue 來保證線程安全了。切記不能在多個線程中共同一個 FMDatabase 對象并且在多個線程中同時使用,這個類本身不是線程安全的,這樣使用會造成數據混亂等問題,我么需要實例化一個 FMDatabaseQueue區吃力多線程下的數據庫操作

封裝FMDatabaseQueue單例類
把FMDatabaseQueue定義成單例的目的是為了保證隊列在內存中是唯一的一個,當有多個數據庫操作任務時,都可以放在同一個隊列中.

單例類繼承自FMDatabaseQueue

#import "FMDB.h"
//繼承FMDatabaseQueue
@interface FMDatabaseQueueManager : FMDatabaseQueue
+ (instancetype)sharedDatabaseQueue;
@end

FMDatabaseQueue
單例類實現

+ (instancetype)sharedDatabaseQueueManager{
 static FMDatabaseQueueManager *instance; 
static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{
// 初始化隊列的操作
 instance = [FMDatabaseQueueManager databaseQueueWithPath:@"/xxx/xxx/xxx/heros.db"];//數據庫路徑
 }); 
return instance;
 }

//實例化單利之后再進行數據庫的操作

//創建數據庫表

+ (void)initialize{
 // 創建和打開數據庫
 [[FMDatabaseQueueManager sharedDatabaseQueue] inDatabase:^(FMDatabase *db) {
 // 建表
 BOOL isCreate = [db executeUpdate:@"create table if not exists t_heros(id integer primary key,name text not null,age integer)"];
 if (isCreate) { 
    NSLog(@"建表成功");
 }
 }];
}
//以新增為例
[[FMDatabaseQueueManager sharedDatabaseQueue] inDatabase:^(FMDatabase *db) { 
       [db executeUpdateWithFormat:@"insert into t_heros(name,age) values('張三',28)"];
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,406評論 6 538
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,034評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,413評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,449評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,165評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,559評論 1 325
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,606評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,781評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,327評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,084評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,278評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,849評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,495評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,927評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,172評論 1 291
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,010評論 3 396
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,241評論 2 375

推薦閱讀更多精彩內容

  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,807評論 18 139
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,692評論 25 708
  • 文/白茶心 自責沒有把新員工用好帶好 李青早早坐在咨詢室里等著圓圓。圓圓到了,坐下喝了口水,沒有說話。她看著墻上的...
    白茶心閱讀 1,044評論 5 3
  • 我心里有一張清晰又模糊的臉,他笑顏懵懂他沉默如斯他眉目清明。那些畫面一點一點從記憶里劃過,好似就是在昨天。 ...
    亦若顏閱讀 419評論 0 6
  • 昨天是個應該被記住的日子 六點多下課我去買了人生第一包紅雙喜 揣著火機在黑漆漆靜悄悄的西操場 抽了人生第一根和第二...
    怪力雪人閱讀 501評論 0 0