iOS数据库存储之FCModel和FMDB

最近看了FCModel和FMDB的源码,觉得FCModel确实不错,但网上资源不多,因此写下这篇文章,来介绍一下FCModel及自己的理解。

一. FMDB

FMDB是对Sqlite的OC的封装,封装了Sqlite的繁琐的C语言代码,对外提供一些简单的接口,用于操作数据库。FMDB的使用参考唐巧老师的博客,写的很详细 在iOS开发中使用FMDB

二. FCModel

  1. FCModel在FMDB的基础上,封装了FMDB提供的功能,加入了对象数据模型。它的readme这样介绍的:FCModel是一个操作SQL易取可选择的核心数据模型,CoreData的替代性选择。适用于那些喜欢Core Data的便利性,但又想要对实现、性能、数据库模式、查询、索引、迁移以及使用原始SQL查询等拥有更多掌控的开发者。
    它是这样做的,比如查询操作,调用FMDB的查询接口之后,利用运行时的class_getProperty方法和KVC机制构造对象,返回给外部OC对象,而不是FMDB直接返回的数据。
  2. CoreData也是支持对象的存储的,能够存储OC对象并且从数据库中获取OC对象,它的底层数据库用的是Sqlite。但是CoreData非线程安全,多线程协作很麻烦。FCModel会有内存级的缓存,可以保证多个页面引用的其实是同一个model,保证数据的一致性。FCModel是用信号量机制来控制线程的同步的。
  3. 下面分析FCModel的源码:
    1)FCModel文件夹下包括FCModel、FCModelCacheObject、FCModelDatabaseQueue三个类的.h .m文件。FCModel继承NSObject,在FMDB的基础上,能够更加方便的在数据库中操作自己的对象。
    2)FCModel类的主要属性及核心方法作用解释如下图:
FCModel主要的属性和方法@2x.png
+ (NSArray *)allLoadedInstances;
当你使用它的时候,如果其他线程加载了一个新的对象,这个array则已经过期了。这个数组中的东西都是被加载的对象,但是不能保证包括了所有的对象,如果你在不同的线程中使用SELECTs。

+ (void)inDatabaseSync:(void (^)(FMDatabase *db))block;
你可以在相同的数据库对象上随意使用自己的查询操作,他们都会在FCModel的私有数据库操作队列中同步执行。

+ (void)dataWasUpdatedExternally;数据被外部更新
如果在实例外部的任何FCModel表执行INSERT/UPDATE/DELETE,则这个方法会被调用

创建、插入、更新、删除操作。使用信号量机制,控制多线程同步。

static dispatch_once_t token;
dispatch_once(&token, ^{
    g_instancesReadLock = dispatch_semaphore_create(1);
    g_instances = [NSMutableDictionary dictionary];
});

此处的dispatch_semaphore是GCD用来同步的一种方式,与它相关的共有三个函数,分别是
dispatch_semaphore_create,
dispatch_semaphore_signal,
dispatch_semaphore_wait。

  1. dispatch_semaphore_t dispatch_semaphore_create(long value);传入的参数为long,输出一个dispatch_semaphore_t类型且值为value的信号量。信号量机制控制线程同步,与OS里面的信号量机制相同
  2. long dispatch_semaphore_signal(dispatch_semaphore_t dsema)这个函数会使传入的信号量dsema的值加1;
      dispatch_semaphore_signal的返回值为long类型,当返回值为0时表示当前并没有线程等待其处理的信号量,其处理的信号量的值加1即可。当返回值不为0时,表示其当前有(一个或多个)线程等待其处理的信号量,并且该函数唤醒了一个等待的线程(当线程有优先级时,唤醒优先级最高的线程;否则随机唤醒)。dispatch_semaphore_wait的返回值也为long型。当其返回0时表示在timeout之前,该函数所处的线程被成功唤醒。当其返回不为0时,表示timeout发生。
  3. long dispatch_semaphore_wait(dispatch_semaphore_t dsema, dispatch_time_t timeout);这个函数会使传入的信号量dsema的值减1;
    这个函数的作用是这样的,如果dsema信号量的值大于0,该函数所处线程就继续执行下面的语句,并且将信号量的值减1;如果desema的值为0,那么这个函数就阻塞当前线程等待timeout(注意timeout的类型为dispatch_time_t,不能直接传入整型或float型数),如果等待的期间desema的值被dispatch_semaphore_signal函数加1了,且该函数(即dispatch_semaphore_wait)所处线程获得了信号量,那么就继续向下执行并将信号量减1。如果等待期间没有获取到信号量或者信号量的值一直为0,那么等到timeout时,其所处线程自动执行其后语句。

在FCModel中,其中一个线程加载所有实例的时候,创建一个信号量锁g_instancesReadLock,执行wait操作,信号量减一,

NSMapTable *classCache = g_instances[self];
NSArray *instances = classCache ? [classCache.objectEnumerator.allObjects copy] : [NSArray array];执行signal操作,信号量加1.

创建锁的逻辑,放在dispatch_once{}中,只会执行一次。

查看缓存中是否有该实例,执行wait操作,信号量减一。查看该对象是否在缓存中,是否在DB中,查询结束后执行Signal操作,信号量加一。如果某个线程在执行查询缓存操作,另一个线程想要执行查询或创建操作,都需要等待,当信号量为0时,再对它执行wait操作,会进入阻塞状态,等待信号量的释放。
注册新的实例及移除实例,都使用信号量控制同步。

三. FCModel的使用

官网上有使用介绍FCModel
要自己创建一个Model类,一定要继承FCModel,在自己定义的Model类中加入创建数据库及创建表的逻辑,然后使用FCModel中的数据库方法即可方便的完成数据库表的增删改查。
下面举个列子,使用FMDB存取数据和使用FCModel存取数据,通过对比,就可以发现FCModel的便捷之处。
获取数据库中的用户表信息,从数据库中查询之后,需要用dictionary存储起来,再用dictionary去构造user实体。那如果还有其他的很多表对应的实体,比如:部门、user的联系人等等,获取每一个表的信息时都要像下面这段代码这样做,用dictionary存储数据库返回的结果,再用dictionary数据构造对应的实体。同理,数据库存储的时候也是这样,先将实体转成dictionary,再一一存储。这样是不是很麻烦?太多繁琐的重复逻辑的代码。使用FCModel的话,就是在将Dictionary生成实体的时候,使用运行时,生成对象,这样所有的实体都可以用这一个接口来完成dictionary转实体,是不是减少了很多代码?所以,对于有ORM需求的项目,使用FCModel比使用FMDB要好很多。

- (void)getAllUsers:(Complection )completion
{
    [_dataBaseQueue inDatabase:^(FMDatabase *db) {
    if ([_database tableExists:usersTableName])
    {
        NSMutableArray* array = [[NSMutableArray alloc] init];
        NSString* sqlString = [NSString stringWithFormat:@"SELECT * FROM %@ ", usersTableName];
        FMResultSet* result = [_database executeQuery:sqlString];
        UserEntity* user = nil;
        while ([result next])
        {
            user = [self userFromResult:result];
            [array addObject:user];
        }
    }
 }];
}
- (UserEntity*)userFromResult:(FMResultSet*)resultSet
{
    NSMutableDictionary *dic = [NSMutableDictionary new];
    [dic safeSetObject:[resultSet stringForColumn:@"name"] forKey:@"name"];
    [dic safeSetObject:[resultSet stringForColumn:@"id"] forKey:@"userId"];
    [dic safeSetObject:[NSNumber numberWithInt:[resultSet intForColumn:@"Sex"]] forKey:@"sex"];
    [dic safeSetObject:[resultSet stringForColumn:@"telphone"] forKey:@"telphone"];
    [dic safeSetObject:[resultSet stringForColumn:@"email"] forKey:@"email"];
    UserEntity* user = [UserEntity dicToUserEntity:dic];
    return user;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,313评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,369评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,916评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,333评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,425评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,481评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,491评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,268评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,719评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,004评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,179评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,832评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,510评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,153评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,402评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,045评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,071评论 2 352

推荐阅读更多精彩内容

  • 2016年国庆假期终于把此书过完,整理笔记和体会于此。 关于书名 书名源于俄罗斯的演员斯坦尼斯拉夫斯基创作的《演员...
    李剑飞的简书阅读 7,229评论 2 65
  • Managing Units of Work(管理工作单位) 调度块允许您直接配置队列中各个工作单元的属性。它们还...
    edison0428阅读 7,955评论 0 1
  • 2007年的10月,我终于有了第一辆属于自己的车,一台红色的甲壳虫。其实买车的时候也很纠结,当时店里只有黑色,蓝色...
    含月6666阅读 437评论 0 0
  • 振和QQ1020833376
    ss谁说的阅读 188评论 0 0
  • 昨天写文章说,我不想被孩子的人生绑架,今天就开始实践了,我让孩子列个学习计划,然后我就心安理得的出去玩了,两个小时...
    苏苏日记阅读 168评论 0 0