FMDB 再封装,多线程安全

一、前言

  • 继之前封装的 FMDB 二次封装,面向模型 ,由于需要多线程操作数据库,之前是针对 FMDatabase 进行封装的,对于多线程操作就很容易造成数据混乱,前文中也有说明。
  • 趁着刚搞完一个小程序项目,有点空余时间,就再封装一个针对 FMDatabaseQueue 的,支持多线程同时操作数据库,避免数据混乱。

二、API设计

考虑封装的统一性,所以API 设计跟 之前封装的 FMDB 二次封装,面向模型 保持一致,当然,以前都是通过返回值来处理,但由于使用 FMDatabaseQueue 回调都是block,因此提供的API 也是block回调,大家可以在Demo 中比对一下

  • 1、单例模式,项目中唯一,方便管理,传入数据库名称,后续操作不同的数据库
/**
 *  @author gitKong
 *
 *  单例创建,项目唯一
 */
+ (instancetype)shareManager:(NSString *)fl_dbName;

  • 2、创表,外界传入指定的类,工具会根据类来创建表,如果此时表已经存在,则跳过,没有才去创建,执行完这个操作后自动关闭数据库,释放内存
#pragma mark -- 创表

/**
 *  @author gitKong
 *
 *  根据类名创建表,如果有则跳过,没有才创建
 
 *  flag:YES表示创建表格操作执行成功 或者 表格已经存在,NO则失败
 
 *  注意:此方法创建表格后不会自动关闭数据库,当开发者执行其他操作(删除数据库除外)后会自动关闭数据库
 
 *  建议使用insert创建表格并添加数据
 */
- (void)fl_createTable:(Class)modelClass complete:(void(^)(FLFMDBQueueManager *manager, BOOL flag))complete;
  • 3、插入数据,可以传入单个模型,或者传入模型数组,此时内部处理了,遍历数组插入的期间数据库不会关闭,直到所有插入完毕后才关闭数据库;同时,如果传入的模型的FLDBID在对应表中已经存在,则执行更新操作,保证FLDBID对应数据的唯一性
#pragma mark -- 插入
/**
 *  @author gitKong
 *
 *  @param model 插入单个模型或者模型数组,如果此时传入的模型对应的FLDBID在表中已经存在,则替换更新旧的
 
 *  flag:YES表示插入数据操作执行成功,NO则失败
 
 *  注意:如果此时没创建表就自动先创建,表名为模型类名,数据插入完毕后会自动关闭数据库
 
 *  建议直接使用insert创建表格并添加数据,因为create方法执行完不会自动关闭数据库
 */
- (void)fl_insertModel:(id)model complete:(void(^)(FLFMDBQueueManager *manager, BOOL flag))complete;
  • 4、查询 提供三个方法
    • (1)、查询表是否存在,执行完毕就会自动关闭数据库,由于此方法存在,框架工具中会出现错误提示信息,因为如果没有对应表,执行操作语句FMDB就会打印出错误信息,当然,所有操作都不需要手动调用,框架中所有操作都有对应的判断
    • (2)、查找指定表中指定FLDBID的单个模型数据
    • (3)、查找指定表中模型数组(所有的)
/**
 *  @author gitKong
 *
 *  查询指定表是否存在
 
 *  flag:YES表示操作执行成功并且 modelClass 表格存在,NO则操作失败或者 modelClass 表格不存在
 
 *  注意:操作执行完毕会自动关闭数据库
 */
- (void)fl_isExitTable:(Class)modelClass complete:(void(^)(FLFMDBQueueManager *manager, BOOL flag))complete;
/**
 *  @author gitKong
 *
 *  查找指定表中指定DBID的模型
 
 *  model:不等于nil,表示查询数据操作执行成功并有数据,返回查询成功的模型数据,nil则表示查询操作失败 或者 查询成功但数据为空 或者 对应的表格不存在
 
 *  注意:操作执行完毕会自动关闭数据库
 */
- (void)fl_searchModel:(Class)modelClass byID:(NSString *)FLDBID complete:(void(^)(FLFMDBQueueManager *manager,id model))complete;
/**
 *  @author gitKong
 *
 *  查找指定表中模型数组(所有的)
 
 *  modelArr:不等于nil,表示查询数据操作执行成功并有数据,返回查询成功的模型数据,nil则表示查询操作失败 或者 查询成功但数据为空 或者 对应的表格不存在
 
 *  注意:操作执行完毕会自动关闭数据库
 */
- (void)fl_searchModelArr:(Class)modelClass complete:(void(^)(FLFMDBQueueManager *manager,NSArray *modelArr))complete;
  • 5、修改 根据指定FLDBID,将新传入的模型替换旧的模型数据,执行完毕后自动关闭数据库
/**
 *  @author gitKong
 *
 *  修改指定DBID的模型
 
 *  flag:YES表示更新操作执行成功,NO则操作失败 或者 对应的表格不存在
 
 *  注意:操作执行完毕会自动关闭数据库
 */

- (void)fl_modifyModel:(id)model byID:(NSString *)FLDBID complete:(void(^)(FLFMDBQueueManager *manager, BOOL flag))complete;
  • 6、删除,此时也提供了三个方法
    • (1)、删除指定表
    • (2)、删除指定表格的所有数据
    • (3)、删除指定表中指定FLDBID的模型
    • (4)、删除数据库
/**
 *  @author gitKong
 *
 *  删除指定表
 
 *  flag:YES表示删除操作执行成功,NO则操作失败 或者 对应的表格不存在
 
 *  注意:操作执行完毕会自动关闭数据库
 */
- (void)fl_dropTable:(Class)modelClass complete:(void(^)(FLFMDBQueueManager *manager, BOOL flag))complete;
/**
 *  @author gitKong
 *
 *  删除指定表格的所有数据
 
 *  flag:YES表示删除操作执行成功,NO则操作失败 或者 对应的表格不存在 或者 没有对应数据可以删除
 
 *  注意:操作执行完毕会自动关闭数据库
 */
- (void)fl_deleteAllModel:(Class)modelClass complete:(void(^)(FLFMDBQueueManager *manager, BOOL flag))complete;
/**
 *  @author gitKong
 *
 *  删除指定表中指定DBID的模型
 
 *  flag:YES表示删除操作执行成功,NO则操作失败 或者 对应的表格不存在 或者 没有对应数据可以删除
 
 *  注意:操作执行完毕会自动关闭数据库
 */
- (void)fl_deleteModel:(Class)modelClass byId:(NSString *)FLDBID complete:(void(^)(FLFMDBQueueManager *manager, BOOL flag))complete;
/**
 *  @author gitKong
 *
 *  删除数据库
 
 *  注意:操作不涉及到数据库操作,如果你通过create创建后执行此操作,不会关闭数据库
 
 *  @return YES 表示删除成功,NO则删除失败
 */
- (BOOL)fl_dropDB;

三、调用

操作和 之前 封装的一样,只是换了种回调方式,之前是通过返回值来判断,此时则通过block 回调中的 flag 来判断操作是否成功,而且操作失败会触犯回滚,详情请看 GitHub Demo,Demo 中作了详细的使用介绍。

[FLFMDBQUEUEMANAGER fl_insertModel:model complete:^(FLFMDBQueueManager *manager, BOOL flag) {
            if (flag) {
                NSLog(@"插入成功");
            }
            else {
                NSLog(@"插入失败");
            }
        }];
__weak typeof(self) weakSelf = self;
[FLFMDBQUEUEMANAGER fl_deleteModel:[FLStudentModel class] byId:model.FLDBID complete:^(FLFMDBQueueManager *manager, BOOL flag) {
        __strong typeof(self) strongSelf = weakSelf;
        if (flag) {
           NSLog(@"删除数据成功");
           [strongSelf showTip:@"删除数据成功"];
        }
       else{
           NSLog(@"删除数据失败");
           [strongSelf showTip:@"删除数据失败"];
       }
}];
__weak typeof(self) weakSelf = self;
[FLFMDBQUEUEMANAGER fl_modifyModel:model byID:model.FLDBID complete:^(FLFMDBQueueManager *manager, BOOL flag) {
                __strong typeof(self) strongSelf = weakSelf;
                if (flag) {
                    [strongSelf showTip:@"修改名字成功"];
                    [strongSelf.tableView reloadData];
                }
                else{
                    [strongSelf showTip:@"修改名字失败"];
                    NSLog(@"修改名字失败");
                }
            }];
 [FLFMDBQUEUEMANAGER fl_searchModel:[FLStudentModel class] byID:textField.text complete:^(FLFMDBQueueManager *manager,id model) {
                __strong typeof(weakSelf) strongSelf = weakSelf;
                FLStudentModel *studentModel = (FLStudentModel *)model;
                
                if (studentModel.FLDBID) {
                    [strongSelf.modelArrM removeAllObjects];
                    [strongSelf.modelArrM addObject:studentModel];
                    [strongSelf.tableView reloadData];
                }
                else{
                    NSLog(@"找不到这个模型");
                    [strongSelf showTip:@"找不到这个模型"];
                }
            }];

四、使用注意:

  • 1、需要在模型中添加一个属性FLDBID,NSString类型,为了绑定对应的数据,从而进行增删改查操作

  • 2、模型中属性不能有 id命名 的属性,会跟sql 的 主键id 冲突。

  • 3、需要插入数据库的模型不支持继承,因为根据类名来创建表,框架内部只能读取当前类的属性,其父类的属性没办法获取。

  • 4、修改数据库中的模型数据只能通过指定的FLDBID作为条件修改。

  • 5、暂时不支持模型属性动态删减,如果删了对应属性(除了FLDBID)不影响使用,但如果增加属性了,只能从新建表存储。

  • 6、嵌套模型暂时不支持单表处理,需要创建多张表处理。

  • 7、由于每次操作完毕都会主动关闭数据库,避免大量数据操作的时候出现文件读写失败问题,因此会在控制台打印 The FMDatabase <FMDatabase: 0x600000094a50> is not open. 不影响实际使用

五、总结

  • 0、为了方便大家使用,提供了两个宏,创建数据库管理实例,其中如果是使用FLFMDBQUEUEMANAGER ,就是使用默认的数据库,文件为gitkong.sqlite,使用 FLFMDBQUEUEMANAGERX 传入名称,就会创建文件为 “名称”.sqlite

    #define FLDB_DEFAULT_NAME @"gitkong"
    
    #define FLFMDBQUEUEMANAGER [FLFMDBQueueManager shareManager:FLDB_DEFAULT_NAME]
    

/**

  • @author gitKong
  • 多数据库操作暂时不开放,有问题需要解决,敬请期待
    */

define FLFMDBQUEUEMANAGERX(DB_NAME) [FLFMDBQueueManager shareManager:DB_NAME]


- 1、取消了断言机制,防止用户误操作而导致程序崩溃,影响用户体验,如果操作失败,可以通过回调中的 `flag` 判断。

- 2、除了查询表是否存在,所有提供的API 中都已经有相应判断,调用者无需手动做判断。

- 3、内部使用 block 回调,但使用时无需担心出现循环引用问题,当然为了确保安全,可以使用下面代码处理。

__weak typeof(self) weakSelf = self;
__weak typeof(self) strongSelf = weakSelf;


- 4、内部实现比较简单,就不作代码分析了,如果大家有什么不明白或者发现什么问题,欢迎留言给我。

- 5、**欢迎大家关注我,喜欢就给个like,随时更新!谢谢支持!** 

# [GitHub地址](https://github.com/gitkong/FLFMDBManager) 欢迎 Star
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,736评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,167评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,442评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,902评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,302评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,573评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,847评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,562评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,260评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,531评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,021评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,367评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,016评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,068评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,827评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,610评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,514评论 2 269

推荐阅读更多精彩内容