FMDB二次封裝工具類,讓你快速學(xué)會(huì)封裝,集成數(shù)據(jù)庫(kù)

上個(gè)版本為了增加用戶體驗(yàn),部分頁(yè)面集成了離線緩存數(shù)據(jù)功能,于是就在項(xiàng)目里使用了數(shù)據(jù)庫(kù)管理離線數(shù)據(jù)。下面交大家一步步學(xué)會(huì)使用FMDB,以及FMDB的二次封裝,同事把我二次封裝的數(shù)據(jù)庫(kù)放出來(lái),希望能夠幫助大家快速學(xué)習(xí),集成數(shù)據(jù)庫(kù)功能吧。

一.首先看一下STDB文件結(jié)構(gòu)

STDB文件結(jié)構(gòu)
  • Table.h主要放一些Table的創(chuàng)建語(yǔ)句, 方便管理我的數(shù)據(jù)庫(kù)各張表創(chuàng)建
  • DBDefine.h主要放一些表名的宏定義,數(shù)據(jù)庫(kù)版本號(hào),數(shù)據(jù)庫(kù)名字等等,方便我們?cè)谑褂脭?shù)據(jù)庫(kù)過(guò)程中更直觀管理版本和各種表
  • STDBTool.h,STDBTool.m具體封裝實(shí)現(xiàn)代碼
Table.h結(jié)構(gòu)
Table.h結(jié)構(gòu)

二.具體實(shí)現(xiàn)功能

1 . STDBTool.h 頭文件看一下

STDBTool.h

我定義了三個(gè)FMDatabaseQueue 因?yàn)閷?shí)際操作中我需要數(shù)據(jù)庫(kù)嵌套,如果只使用一個(gè)FMDatabaseQueue 將會(huì)陷入死循環(huán),下面看一下1FMDatabaseQueue源碼分析一下原因:
FMDatabaseQueue 全局變量
FMDatabaseQueue 全局變量
創(chuàng)建一個(gè)隊(duì)列_queue = dispatch_queue_create([[NSString stringWithFormat:@"fmdb.%@", self] UTF8String], NULL);默認(rèn)是串行隊(duì)列,數(shù)據(jù)庫(kù)操作的時(shí)候FMDB源碼如下圖
FMDatabaseQueue 同步執(zhí)行源碼
FMDatabaseQueue 同步執(zhí)行源碼

同步執(zhí)行串行隊(duì)列 block塊里按著順序執(zhí)行。
demo0
demo0

任務(wù)1執(zhí)行 ——>任務(wù)二等待任務(wù)一執(zhí)行完畢執(zhí)行,任務(wù)一等待任務(wù)二執(zhí)行完畢執(zhí)行,死鎖。 如果新建一個(gè)串行隊(duì)列
demo1
demo1

這樣就沒(méi)有問(wèn)題。 至于同步異步并行串行網(wǎng)上也有很多,不在一一介紹啦。

2 . STDBTool的初始化 ,很顯然 STDBTool用的是單例啦, 看一下alloc方法

STDBTool 初始化

這里面也就是創(chuàng)建一下數(shù)據(jù)庫(kù)文件,設(shè)置數(shù)據(jù)庫(kù)版本號(hào)。
當(dāng)數(shù)據(jù)庫(kù)有新表需要增加時(shí),我們需要改變數(shù)據(jù)庫(kù)版本號(hào),對(duì)數(shù)據(jù)庫(kù)進(jìn)行升級(jí)。

  • .查詢數(shù)據(jù)庫(kù)當(dāng)前版本
NSString * sql = [NSString stringWithFormat:@"PRAGMA user_version"];
FMResultSet * rs = [db executeQuery:sql];  
    int nVersion = 0;  
    while ([rs next]) {  
        nVersion = [rs intForColumn:@"user_version"];
    } 
  • 與宏定義數(shù)據(jù)庫(kù)版本不一致 需要更新 設(shè)置 數(shù)據(jù)庫(kù)版本號(hào)
NSString *sql = [NSString stringWithFormat:@"PRAGMA user_version = %ld",(long)newVersion];  
    BOOL ret = [db executeUpdate:sql];

STDBTool初始化就這些吧。也沒(méi)什么難點(diǎn),主要是跟大家一起回顧一下。

3 .實(shí)現(xiàn)的數(shù)據(jù)庫(kù)數(shù)據(jù)操作功能

一般數(shù)據(jù)庫(kù)操作無(wú)非是增刪改查這些基本操作,當(dāng)然我的封裝也是基于實(shí)現(xiàn)這些功能的。但是根據(jù)具體情況我們需要進(jìn)行封裝。

  1. 執(zhí)行單個(gè)sql語(yǔ)句時(shí)候,不需要使用事務(wù)處理,我們需要知道操作類型,這里我寫了個(gè)枚舉type 便于區(qū)分
-(void)executeSQL:(NSString *)sqlStr actionType:(ST_DB_ActionType)actionType withBlock:(void(^)(BOOL bRet, FMResultSet *rs, NSString *msg))block{
    [_dbQueue inDatabase:^(FMDatabase *db) {
        if (actionType == ST_DB_SELECT) {
            //查詢語(yǔ)句 需要返回記錄集
          FMResultSet * rs = [db executeQuery:sqlStr];
            if ([db hadError]) {
                block(NO,rs,[db lastErrorMessage]);
                NSLog(@"executeSQL error %d:  %@",[db lastErrorCode],[db lastErrorMessage]);
            }else{
                block(YES,rs,nil);
            }
        }else{
            //更新操作 只關(guān)心操作是否執(zhí)行成功,不關(guān)心記錄集  返回布爾值  無(wú)執(zhí)行結(jié)果
            BOOL ret = [db executeUpdate:sqlStr];
            if ([db hadError]) {
                block(NO,nil,[db lastErrorMessage]);
                NSLog(@"executeSQL error %d:  %@",[db lastErrorCode],[db lastErrorMessage]);
            }else{
                block(ret,nil,nil);
            }
        }
    }];
}
  1. 根據(jù)查詢結(jié)果 確定是更新還是新增操作,只需要知道是否操作成功,不關(guān)心結(jié)果集 只處理一個(gè)查詢更新,不需要事務(wù)處理
- (void)executeRelevanceSql:(NSArray *)sqlList withBlock:(void(^)(BOOL ret,NSString * errMsg))block{
    __block BOOL ret;
    [_dbQueue inDatabase:^(FMDatabase *db) {
        FMResultSet * rs = [db executeQuery:sqlList[0]];
        if ([db hadError]) {
            block(NO,[db lastErrorMessage]);
            NSLog(@"da_error_%@",[db lastErrorMessage]);
        }
        
        int nCount = 0;
        if ([rs next]) {
            //獲取查詢數(shù)據(jù)的個(gè)數(shù)
            nCount = [rs intForColumnIndex:0];
        }
        [rs close];
        
        NSString * nextSqlString = nil;
        if (nCount > 0) {
            //查詢到了結(jié)果  執(zhí)行update操作
            nextSqlString = sqlList[1];
        }else{
            //查詢無(wú)結(jié)果  執(zhí)行 insert into 操作
            nextSqlString = sqlList[2];
        }
        
        ret = [db executeUpdate:nextSqlString];
        if ([db hadError]) {
            block(NO,[db lastErrorMessage]);
            NSLog(@"da_error_%@",[db lastErrorMessage]);
        }else{
            block(ret, nil);
        }
    }];
}

注:sql語(yǔ)句數(shù)組,sqlList[0]查詢select語(yǔ)句 sqList[1] update更新語(yǔ)句 sqlList[2] insert into 插入語(yǔ)句

  1. 根據(jù)查詢結(jié)果 確定是更新還是新增操作,只需要知道是否操作成功,不關(guān)心結(jié)果集 只處理一個(gè)查詢更新,不需要事務(wù)處理
- (void)executeRelevanceSql:(NSArray *)sqlList withBlock:(void(^)(BOOL ret,NSString * errMsg))block{
    __block BOOL ret;
    [_dbQueue inDatabase:^(FMDatabase *db) {
        FMResultSet * rs = [db executeQuery:sqlList[0]];
        if ([db hadError]) {
            block(NO,[db lastErrorMessage]);
            NSLog(@"da_error_%@",[db lastErrorMessage]);
        }
        int nCount = 0;
        if ([rs next]) {
            //獲取查詢數(shù)據(jù)的個(gè)數(shù)
            nCount = [rs intForColumnIndex:0];
        }
        [rs close];
        
        NSString * nextSqlString = nil;
        if (nCount > 0) {
            //查詢到了結(jié)果  執(zhí)行update操作
            nextSqlString = sqlList[1];
        }else{
            //查詢無(wú)結(jié)果  執(zhí)行 insert into 操作
            nextSqlString = sqlList[2];
        }
        ret = [db executeUpdate:nextSqlString];
        if ([db hadError]) {
            block(NO,[db lastErrorMessage]);
            NSLog(@"da_error_%@",[db lastErrorMessage]);
        }else{
            block(ret, nil);
        }
    }];
}

注: sql語(yǔ)句數(shù)組,sqlList[0]查詢select語(yǔ)句 sqList1update更新語(yǔ)句 sqlList2 insert into 插入語(yǔ)句

4 . sqlList 是一個(gè)二維數(shù)組,每一個(gè)成員包含三個(gè)sql語(yǔ)句,分別是查詢,更新,插入,并且根據(jù)查詢結(jié)果返回是執(zhí)行更新 還是 插入操作。使用dbQueue2 用于直接調(diào)用。批量處理,使用事務(wù)。

- (void)executeDbQueue2RelevanceTransactionSqlList:(NSArray *)sqlList withBlock:(void(^)(BOOL bRet, NSString *msg, BOOL *bRollback))block{
    __block BOOL ret = NO;
    [_dbQueue2 inTransaction:^(FMDatabase *db, BOOL *rollback) {
        for (NSArray * singleSqlList in sqlList ) {
            FMResultSet * rs = [db executeQuery:singleSqlList[0]];
            if ([db hadError]) {
                block(NO,[db lastErrorMessage],rollback);
                NSLog(@"da_error_%@",[db lastErrorMessage]);
            }else{
                int nCount = 0;
                while ([rs next]){
                    nCount  = [rs intForColumnIndex:0];
                }
                [rs close];
                
                NSString * nextSqlString = nil;
                if (nCount > 0){
                    //執(zhí)行更新
                    nextSqlString = singleSqlList[1];
                }
                else{
                    //執(zhí)行插入
                    nextSqlString = singleSqlList[2];
                }
                
                 ret = [db executeUpdate:nextSqlString];
                if ([db hadError])
                {
                    block(NO, [db lastErrorMessage], rollback);
                    NSLog(@"executeSql Err %d: %@", [db lastErrorCode], [db lastErrorMessage]);
                }
            }
        }
         block(ret, nil, rollback);
    }];
}

注:sql語(yǔ)句數(shù)組,sqlArr[i][0]:查詢語(yǔ)句;sqlArri:update語(yǔ)句;sqlArri:insert into語(yǔ)句

5.sql語(yǔ)句數(shù)組中每個(gè)成員有2條語(yǔ)句,第一條是select語(yǔ)句,第二條是insert into語(yǔ)句,根據(jù)第一個(gè)sql的執(zhí)行結(jié)果確定第二條語(yǔ)句是否執(zhí)行。根據(jù)查詢結(jié)果確定是否新增,批量處理,使用事務(wù)處理,不需要返回記錄集使用dbQueue2,用于程序中直接調(diào)用(非封裝在其他方法中)

-(void)executeInsertTransactionSqlList:(NSArray *)sqlStrList withBlock:(void(^)(BOOL bRet, NSString *msg, BOOL *bRollback))block
{
    __block BOOL ret = NO;
     NSLog(@"開始啦---");
    [_dbQueue2  inTransaction:^(FMDatabase *db, BOOL *rollback){
        
        for (NSArray *sqlArray in sqlStrList){
            FMResultSet *rs = [db executeQuery:[sqlArray objectAtIndex:0]];
            if ([db hadError]){
                block(NO, [db lastErrorMessage], rollback);
                NSLog(@"executeSql Err %d: %@", [db lastErrorCode], [db lastErrorMessage]);
            }
            
            int nCount = 0;
            while ([rs next]){
                nCount = [rs intForColumnIndex:0];
            }
            [rs close];
            
            if (nCount <= 0){
                ret = [db executeUpdate:[sqlArray objectAtIndex:1]];
                if ([db hadError])
                {
                    block(NO, [db lastErrorMessage], rollback);
                    NSLog(@"executeSql Err %d: %@", [db lastErrorCode], [db lastErrorMessage]);
                }
            }
        }
        block(ret, nil, rollback);
    }];
}

注:sql語(yǔ)句數(shù)組,sqlArr[i][0]:查詢語(yǔ)句;sqlArri:insert into語(yǔ)句
6.批量處理更新或者新增sql語(yǔ)句,不需要返回記錄集 無(wú)事務(wù)處理

- (void)executeSQLList:(NSArray *)sqlStrList db:(FMDatabase *)db withBlock:(void(^)(BOOL bRet, NSString *msg))block{
    __block BOOL bRet = NO;
    for (NSString * sqlString in sqlStrList) {
        bRet = [db executeUpdate:sqlString];
        if ([db hadError]) {
            block(bRet,[db lastErrorMessage]);
            NSLog(@"executeSQLList Err %d: %@", [db lastErrorCode], [db lastErrorMessage]);
            break;
        }
    }
    block(bRet,nil);
}

注: sql語(yǔ)句數(shù)組update或者insert into語(yǔ)句

7.批量處理更新或者新增sql語(yǔ)句,并且不需要返回記錄集,使用事務(wù)處理

-(void)executeTransactionSqlList:(NSArray *)sqlStrList withBlock:(void(^)(BOOL bRet, NSString *msg, BOOL *bRollback))block
{
    __block BOOL bRet = NO;
    [_dbQueue  inTransaction:^(FMDatabase *db, BOOL *rollback){
        
        for (NSString *sqlStr in sqlStrList)
        {
            bRet = [db executeUpdate:sqlStr];
            if ([db hadError])
            {
                block(bRet, [db lastErrorMessage], rollback);
                NSLog(@"executeSQLList Err %d: %@", [db lastErrorCode], [db lastErrorMessage]);
                break;
            }
        }
        block(bRet, nil, rollback);
    }];
}

注:sql語(yǔ)句數(shù)組update或者insert into語(yǔ)句

還有幾個(gè)方法在防止死循環(huán)嵌套 類似的函數(shù)。
github地址可以直接下載使用,感覺(jué)有用的話star一下,謝謝大家。希望能幫到大家。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,345評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,494評(píng)論 3 416
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,283評(píng)論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,953評(píng)論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,714評(píng)論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,186評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,255評(píng)論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,410評(píng)論 0 288
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,940評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,776評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,976評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,518評(píng)論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,210評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,642評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,878評(píng)論 1 286
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,654評(píng)論 3 391
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,958評(píng)論 2 373

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