美文网首页I love iOS代码片段iOS开发记录
iOS-FMDB改进方案YIIFMDB:直接操作Model,纯面

iOS-FMDB改进方案YIIFMDB:直接操作Model,纯面

作者: 狼居胥侯 | 来源:发表于2018-10-29 17:35 被阅读154次

    我在写UDUserDefaultsModel(文章链接,github)这个库时曾经立下一个flag:要写一个基于model来存取数据库的库,最近刚离职,所以就整合了一下,希望大家多多支持。

    在iOS开发过程当中,难免用到数据库,以FMDB居多。以下是一个根据年龄筛选数据的sql语句:

    select * from Student where age > 10 and age < 20 or age > 30 order by age desc limit 20
    

    这样写其实没什么问题,但是在我个人看来难以接受,字符串看起来太别扭。比如我再添加一个条件,那么就需要修改整个字符串了。

    如果可以很好的控制sql语句,将大大提高编程效率。为此,YIIFMDB就改善了这个缺陷:纯面向对象,直接操作Model,完全不需要写sql语句

    其灵感源自于php的Yii 2架构,因为我在看php代码当中,我发现根本就看不到sql语句,而php程序猿也说:“我们的工作就是操作数据库,但是却不写sql语句”。

    以下是YIIFMDB详细用法:
    YIIFMDB有两个类:YIIFMDB和YIIParameters。其中YIIFMDB封装了数据库相关的操作,比如增删改查之类,而YIIParameters则封装了where之后的参数,比如上段代码当中的:

    age > 10 and age < 20 or age > 30 order by age desc limit 20
    

    就可以在YIIParameters当中完成。

    接下来逐一介绍YIIParameters类和YIIFMDB类的使用:

    YIIParameters类

    sql语句当中where之后的参数基本上由以下模块构成:

    • and(与操作)
    • or(或操作)
    • order by(排序)
    • limit(数量限制)

    其中的andor又要配置“>,<,=,>=,<=,!=,like”关系,order by又有“ase,dese”的排序操作。

    YIIParameters这个类就包含了以上所有元素。以上面的where之后的sql语句为例,具体用法如下:

    // 初始化YIIParameters
    YIIParameters *parameters = [[YIIParameters alloc] init];
    // 执行and操作,将age限制在10-20之间
    // age > 10,YIIParametersRelationTypeGreaterThan标志">"
    [parameters andWhere:@"age" value:@"10" relationType:YIIParametersRelationTypeGreaterThan];  
    // age < 20,YIIParametersRelationTypeLessThan标志"<"
    [parameters andWhere:@"age" value:@"20" relationType:YIIParametersRelationTypeLessThan];
    // 以上是and,也就是形成的sql语句为: age > 10 and age < 20
    // 执行or操作,将age限制在age > 30 以上
    [parameters orWhere:@"age" value:@"30" relationType:YIIParametersRelationTypeGreaterThan];
    // 根据age进行降序排列
    // YIIParametersOrderTypeDesc表示降序"desc",YIIParametersOrderTypeAsc
    [parameters orderByColumn:@"age" orderType:YIIParametersOrderTypeDesc];
    // 将数据的个数限制在20个
    parameters.limitCount = 20;
    

    配置完毕,验证其是否配置正确,那么可以调用一下方法就行了:

    NSLog(@"where参数为:%@", parameters.whereParameters);
    

    当然,如果参数都没法配置了,则可以设置whereParameters。而对于YIIParameters更详细的解释请参考 YIIFMDB中的YIIParameters.h

    YIIFMDB

    YIIParameters用来配置sql语句当中where之后的参数,而YIIFMDB类则是对数据库操作的进一步封装,具体如下:

    获取YIIFMDB单例

    YIIFMDB *db = [YIIFMDB shareDatabase]; // 推荐使用
    // 或者
    YIIFMDB *db = [YIIFMDB shareDatabaseForName:@"ABC.sqlite" path:path]; // 自定义数据库名字和路径,在第一次实例的时候传入,以后使用上面方法即可。
    

    主键的字段

    @property (nonatomic, readonly, copy) NSString *primaryKey; // 返回"yii_pkID",我自己在创建数据库是配置的主键字段

    是否打印log

    @property (nonatomic, assign) BOOL shouldOpenDebugLog; // 默认为NO,设为YES,会在控制器后台打印数据库操作相关的一些信息

    创建一张表

    [[YIIFMDB shareDatabase] createTableWithModelClass:[LCPersonModel class] excludedProperties:nil tableName:@"Person"];
    

    此方法是创建一张名为@"Person"表,并且,表里面的字段也就是LCPersonModel里面的属性,字段的数据类型也对应LCPersonModel里面的数据类型

    插入一条数据(增)

      LCPersonModel *model = [[LCPersonModel alloc] init];
      model.name = [NSString stringWithFormat:@"lc%d", (arc4random() % 100)];
      model.gender = arc4random() % 2;
      model.age = arc4random() % 80;
      model.floatNumber = (arc4random() % 20) / 100.0;
      model.doubleNumber = (arc4random() % 20) / 100.0;
      model.isMan = arc4random() % 2;
      model.number = @(arc4random() % 10);
                
      YIIFMDB *db = [YIIFMDB shareDatabase];
      BOOL isSuccess = [db insertWithModel:model tableName:tableName];  //插入一条数据
      [db insertWithModels:@[model] tableName:tableName];     // 批量插入数据
    

    删除数据(删)

    -(BOOL)deleteFromTable:(NSString * _Nonnull)tableName whereParameters:(YIIParameters *)parameters; // 根据参数删除一条数据,YIIParameters参考上面
    -(BOOL)deleteAllDataFromTable:(NSString * _Nonnull)tableName; // 删除表中的所有数据

     YIIFMDB *db = [YIIFMDB shareDatabase];
     YIIParameters *parameters = [[YIIParameters alloc] init];
     // db.primaryKey 是数据库的主键,这条语句意思是删除主键 = 1的那条数据
     [parameters andWhere:db.primaryKey value:@"1" relationType:YIIParametersRelationTypeEqualTo];
     [db deleteFromTable:tableName whereParameters:parameters];
    

    更改数据(改)

    -(BOOL)updateTable:(NSString * _Nonnull)tableName dictionary:(NSDictionary * _Nonnull)dictionary whereParameters:(YIIParameters *)parameters; // 更新一条数据

      YIIFMDB *db = [YIIFMDB shareDatabase];
      YIIParameters *parameters = [[YIIParameters alloc] init];
      // 参数设置为主键 = 10
      [parameters andWhere:db.primaryKey value:@"10" relationType:YIIParametersRelationTypeEqualTo];
      // 将主键为10的那条数据的name更改为monkey
      [db updateTable:tableName dictionary:@{@"name": @"monkey"} whereParameters:parameters];
    

    查询数据

    -(NSArray *)queryFromTable:(NSString * _Nonnull)tableName model:(Class _Nonnull)modelClass whereParameters:(YIIParameters *)parameters; // 根据YIIParameters条件从表为tableName的查询数据

      YIIFMDB *db = [YIIFMDB shareDatabase];
      YIIParameters *parameters = [[YIIParameters alloc] init];
      [parameters andWhere:db.primaryKey value:@"5" relationType:YIIParametersRelationTypeLessThan];
      NSLog(@"主键小于5的数据:%@", [db queryFromTable:tableName model:[LCPersonModel class] whereParameters:parameters]);
    

    除了增删改查之外,YIIFMDB还提供了增加一个属性,删除一张表,获取表中所有字段名,获取表中数据个数,表是否存在,求和,求平均值,最大值,最小值等功能,详情请参考YIIFMDB的文档。

    线程安全操作(队列和事务)

    由于FMDB本身就是是不安全的,上面的方法也是不安全的,为了保证其安全则需要结合队列和事务操作,参考FMDB的队列和事务。

    -(void)inDatabase:(dispatch_block_t)block; // 将数据库相关操作写在block里可保证线程安全

      YIIFMDB *db = [YIIFMDB shareDatabase];
       [db inDatabase:^{
           // 增删改查放在此代码块里执行则可以保证线程安全
       }];
    

    -(void)inTransaction:(void(^)(BOOL *rollback))block; // 在block里写入代码可执行回滚操作

      YIIFMDB *db = [YIIFMDB shareDatabase];
      [db inTransaction:^(BOOL *rollback) {
      // 如果某一个操作失误,则可以执行回滚操作
      BOOL isSuccess = YES;   // 数据库操作是否操作成功
      if (!isSuccess) {
          *rollback = YES;  //  回滚操作
                        
          return ;
         }
      }];
    

    这里还有两个缺陷:

    • 未支持联表查询
    • 未支持Model当中套Model的插入。

    如果是联表查询的话,那么需要获取到YIIFMDB单例的"currentDatabase"来实现联表查询,而Model套Model的,则最好是创建两张表。

    目前YIIFMDB已支持Cocoapods,地址:Github。(不要吝啬你的star)

    相关文章

      网友评论

      • 74e11af8526c:这个不错啊,还有的功能都有了,而且YIIParameters还自己管理sql语句,漂亮
      • 玉思盈蝶:现在离职,厉害了呀:smile:

      本文标题:iOS-FMDB改进方案YIIFMDB:直接操作Model,纯面

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