美文网首页项目以及封装iOS Mark转载
FMDB 再封装,多线程安全

FMDB 再封装,多线程安全

作者: gitKong | 来源:发表于2017-01-15 17:56 被阅读1647次

    一、前言

    • 继之前封装的 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

    相关文章

      网友评论

      • boxKUn:一直插入数据失败,是什么回事
        DB Query: INSERT INTO Model (FLDBID, PassWord, CreateTime, LastLoginTime) VALUES ('123456', '999999', 2018-09-08 10:57:43 +0000, 2018-09-08 10:57:43 +0000);
        2018-09-08 18:57:43.698981+0800 TestDemo[1260:285690] DB Path: /var/mobile/Containers/Data/Application/4671D55F-E4F6-4B64-8547-FD4FE859CBF1/Documents/gitkong.sqlite
        2018-09-08 18:57:43.699523+0800 TestDemo[1260:285652] 插入数据失败
        2018-09-08 18:57:43.700876+0800 TestDemo[1260:285690] Error calling sqlite3_step (1: cannot rollback - no transaction is active) SQLITE_ERROR
        gitKong:时间你传时间戳试试
      • jackyding:你好,因为项目需要,看到了你写的,功能正好是我们想要的,所以基于你的功能增加了表字段的动态增加和减少,你的代码也重构了,加你QQ了340004514,有时间直接聊
      • Zz7777777:FLFMDBQueueManager *manager = [FLFMDBQueueManager shareManager:@"HZORM"];
        Person *p = [[Person alloc] init];
        p.FLDBID = @"1";
        p.userName = @"张三";
        p.userPassWord = @"1234556";
        p.userImageURL = @"www.baidu.com";
        [manager fl_insertModel:p complete:^(FLFMDBQueueManager *manager, BOOL flag) {

        }];

        The FMDatabase <FMDatabase: 0x1700bde20> is not open.

        老是抱数据库未打开
      • 本僧先僧:楼主,请教个问题,fmdb能存嵌套对象吗?就是对象里面的属性包含有对象,这种fmdb能存吗
        gitKong:@鎏星蝴蝶飞 可以,可以看demo
      • 屈涯:里面有内存泄露。 而且没有使用 FMDatabaseQueue 会导致数据库异常
        屈涯:@gitKong 插入 可以加一个只根据表名插入。 只根据表名查询的
        屈涯:@gitKong 嗯. FMDatabaseQueue的写的不错, 有检测过内存的问题吗?。我把原来的FMDatabase换成FMDatabaseQueue
        gitKong:@苏_沫 有两个的,一个就是用queue
      • gitKong:哦哦,我没注意:stuck_out_tongue:谢谢
      • no9xavier:问题来了,strongSelf 到底是 __weak 还是 __strong?
        manajay:@no9xavier 可以的 兄弟
        no9xavier:@gitKong 可是你文章里面写的是 __weak typeof(self) strongSelf = weakSelf
        gitKong:strong
      • Dream1213:哥们 集合数据 取出来 便利会崩溃
        Dream1213:我刚才 看了下您写的 库。 你查出来直接 字符串 类型数据 我遍历 会崩溃数据类型不对
        Dream1213:您还在么?
        gitKong:@swp_song 你确定取出来的不是nil?
      • First灬DKS:请问博主:如果手机更换了登陆的账号,此时这个还可以使用吗?
        gitKong:@First灬DKS 会的,所以切换账号后,建议你清空数据库
        First灬DKS:@gitKong 如果A的账号已经有缓存数据了,此时更换成B的账号,那么B从缓存中会获取到A的那些数据吗?
        gitKong:@First灬DKS 可以,因为数据库存在本地的

      本文标题:FMDB 再封装,多线程安全

      本文链接:https://www.haomeiwen.com/subject/uxnybttx.html