EasyDataKit 简单易用的 SQLite 轮子
<p>EasyDataKit 是一个基于 FMDB 封装的库,它可以免去开发者编写复杂 SQL 的困扰,更加专注业务上的事情,从而提高开发效率。</p> <h2><strong>特征</strong></h2> <ul> <li>类 ORM 接口</li> <li>自动创建库和表,支持表新增字段的修改</li> <li>支持 where 查询语句</li> <li>自动事务提升插入效率</li> </ul> <h2><strong>使用</strong></h2> <p>EasyDataKit 适用于将网络请求的数据持久化到数据库中,特别是处理网络请求数据时不习惯把数据转换成 model。笔者的浅见是:转换对性能是有消耗的,获得的可读性好处也可以通过字符串常量解决。</p> <p>假设你通过网络请求获取到了数据:</p> <pre> <code class="language-objectivec">{ "data": { "id": "56d177a27cb3331100465f72", "messagePrefix": "饭否每日精选", "content": "饭否每日精选", "topicId": 1345, "briefIntro": "饭否是国内著名的小众轻博客社区,氛围独特,清新自由。关注饭否每日精选,看看尘嚣之外,大家谈论什么。", "keywords": "饭否 精选 短博客 社区", "timeForRank": "2016-02-27T11:06:30.731Z", "lastMessagePostTime": "2016-11-06T02:42:52.111Z", "topicPublishDate": "2016-02-26T16:00:00.000Z", "createdAt": "2016-02-27T10:17:06.295Z", "updatedAt": "2016-11-01T04:30:08.973Z", "subscribersCount": 1207100, "subscribedStatusRawValue": 1, "subscribedAt": "2016-10-18T09:57:24.424Z", "rectanglePicture": { "thumbnailUrl": "https://cdn.ruguoapp.com/o_1ach3c6o011j91ljjtmdhlhnffo.jpg?imageView2/1/w/120/h/180", "middlePicUrl": "https://cdn.ruguoapp.com/o_1ach3c6o011j91ljjtmdhlhnffo.jpg?imageView2/1/w/200/h/300", "picUrl": "https://cdn.ruguoapp.com/o_1ach3c6o011j91ljjtmdhlhnffo.jpg?imageView2/0/h/1000", "format": "png" }, "squarePicture": { "thumbnailUrl": "https://cdn.ruguoapp.com/o_1ach6nm599m94re1gvj14r71jaso.jpg?imageView2/0/w/120/h/120", "middlePicUrl": "https://cdn.ruguoapp.com/o_1ach6nm599m94re1gvj14r71jaso.jpg?imageView2/0/w/300/h/300", "picUrl": "https://cdn.ruguoapp.com/o_1ach6nm599m94re1gvj14r71jaso.jpg?imageView2/0/h/1000", "format": "png" }, "pictureUrl": "https://cdn.ruguoapp.com/o_1ach3c6o011j91ljjtmdhlhnffo.jpg?imageView2/1/w/200/h/300", "thumbnailUrl": "https://cdn.ruguoapp.com/o_1ach6nm599m94re1gvj14r71jaso.jpg?imageView2/0/w/300/h/300" } } </code></pre> <p>你可将这段 JSON String 转换成 Dictionary 或 Array:</p> <pre> <code class="language-objectivec">NSDictionary *dictionary = [NSJSONSerializationJSONObjectWithData:dataoptions:NSJSONReadingAllowFragmentserror:nil]; NSDictionary *subscribe = dictionary[@"data"]; </code></pre> <p>接着便可使用 EasyDataKit 的 API 进行存储:</p> <pre> <code class="language-objectivec">EDKEntity *subscribeEntity = [[EDKEntity alloc]initWithTableName:@"subcribes"dbName:nil]; [subscribeEntitysaveData:subscribeprimaryColumn:@"id"relationShip:nilindexes:nil]; </code></pre> <p>可以手动为数据添加列,实现满足业务的需求:</p> <pre> <code class="language-objectivec">NSMutableDictionary *subcribeInfo = [[NSMutableDictionary alloc]initWithDictionary:subscribe]; [subcribeInfosetObject:<a href="http://www.jobbole.com/members/yaowei729">@1</a>forKey:@"isSubcribed"]; EDKEntity *subscribeEntity = [[EDKEntity alloc]initWithTableName:@"subcribes"dbName:nil]; [subscribeEntitysaveData:subcribeInfoprimaryColumn:@"id"relationShip:nilindexes:nil]; </code></pre> <p>如果你想让某纪录关联其它对象,可以将对象存储后返回的 id 作为 value,key 是该纪录原本对应该对象的字段,这相当于用 id 这个值去替换原本字段对应的对象,从而达到拆分的目的:</p> <pre> <code class="language-objectivec">id rowId = [rectanglePictureEntitysaveData:subscribe[@"rectanglePicture"]primaryColumn:nilrelationShip:nil]; EDKEntity *subscribeEntity = [[EDKEntity alloc]initWithTableName:@"subcribes"dbName:nil]; [subscribeEntitysaveData:subscribeprimaryColumn:@"id"relationShip:@{@"rectanglePicture": rowId}indexes:nil]; </code></pre> <p>EasyDataKit 还可以存储索引</p> <pre> <code class="language-objectivec">NSDictionary *subcribeInfo = [[NSDictionary alloc]initWithDictionary:subscribe]; EDKEntity *subscribeEntity = [[EDKEntity alloc]initWithTableName:@"subcribes"dbName:@"TestIndex"]; [subscribeEntitysaveData:subcribeInfoprimaryColumn:nilrelationShip:nilindexes:@[@[@"topicId"], @[@"content", @"messagePrefix"]]]; </code></pre> <p>对存储来说,EasyDataKit 还提供了自动 ALTER TABLE 添加列的功能,方便开发者应对升级,原理是当检测到待存储的字典 keys 数组元素个数比之前已经在表中的列多时,则会自动为表添加新的列,即并不支持修改列和删除列的操作,而 EasyDataKit 对创建的索引是支持修改删除的。</p> <p>查询:</p> <pre> <code class="language-objectivec">// select by id EDKEntity *entity = [[EDKEntity alloc]initWithTableName:@"messages"dbName:nil]; id object = [entityqueryByPrimaryKey:@"581d2fdb36a4471100e311d6"withColumns:@[@"topicId", @"commentCount", @"topic"]]; NSLog(@"%@", object); NSArray *objects = [entityqueryWithColumns:nilwhere:@"WHERE commentCount < ? and read = ?"arguments:@[@20, @1]]; NSLog(@"%@", objects); </code></pre> <p>查询嵌套对象并将其转换为 Dictionary 或 Array:</p> <pre> <code class="language-objectivec">EDKEntity *subcribeEntity = [[EDKEntity alloc]initWithTableName:@"subcribes"dbName:nil]; // select by id id subcribe = [subcribeEntityqueryByPrimaryKey:@"56d177a27cb3331100465f72"withColumns:@[@"squarePicture"]]; // subcribe is a json string NSData *data = [subcribe[@"squarePicture"]dataUsingEncoding:NSUTF8StringEncoding]; NSError *error; NSDictionary *jsonDict = [NSJSONSerializationJSONObjectWithData:dataoptions:0error:&error]; NSLog(@"JSONDict: %@", jsonDict); </code></pre> <p>更新:</p> <pre> <code class="language-objectivec">EDKEntity *entity = [[EDKEntity alloc]initWithTableName:@"messages"dbName:nil]; [entityupdateByPrimaryKey:@"5805905a319a9c1200833660"set:@{@"read": @"0", @"commentCount": @99}]; [entityupdateWithSet:@{@"messageId": @"2333333"}where:@"WHERE commentCount > ?"arguments:@[@50]]; </code></pre> <p>删除:</p> <pre> <code class="language-objectivec">EDKEntity *entity = [[EDKEntity alloc]initWithTableName:@"messages"dbName:nil]; // delete by id [entitydeleteByPrimaryKey:@"5805905a319a9c1200833660"]; // delete by where [entitydeleteWithWhere:@"WHERE popularity = ?"arguments:@[@"93"]]; // delete all [entitydeleteAll]; </code></pre> <pre> <code class="language-objectivec">EDKEntity *entity = [[EDKEntity alloc]initWithTableName:@"messages"dbName:nil]; // delete by id [entitydeleteByPrimaryKey:@"5805905a319a9c1200833660"]; // delete by where [entitydeleteWithWhere:@"WHERE popularity = ?"arguments:@[@"93"]]; // delete all [entitydeleteAll]; </code></pre> <p>由上面可以看出,只要创建出 EDKEntity 对象,就可以轻松加愉快地进行存储,查找,修改,删除操作。开发者无需创建数据库、表,EasyDataKit 也支持 db 的划分,但不提供内存缓存,原因是笔者认为没有热块的数据库缓存意义不是太大。当有写操作发生的时候,EasyDataKit 会通过轮询的事务机制打包写操作,从而提高频繁写操作的效率。(更详细的使用请移步 EasyDataKit 中的 Example)</p> <h2><strong>源码简析</strong></h2> <p>EasyDataKit 有个 swizzle 了 NSMutableDictionary 的 setObject:forKey: 和 NSMutableArray 的 addObject:,使得应对空值不会 crash。</p> <p>EasyDataKit 还有个递归方法:dealWithObject:,主要有两个用途:一是用来检测除了 NSDictionary、NSArray、NSString、NSNumber、NSNull 这些类型以外的合法性,譬如 UIView 类型直接调用 description 转换成字符串;二是为了能让嵌套的 Dictionary 或 Array 以 JSON String 的形式存入数据库,在取出后仍可以将其转换回 Dictionary 或 Array。</p> <p>DISPATCH_SOURCE_TYPE_TIMER 创建的定时器。</p> <p> </p> <p> </p> <p>来自:http://ios.jobbole.com/90509/</p> <p> </p>
本文由用户 热血pk007 自行上传分享,仅供网友学习交流。所有权归原作者,若您的权利被侵害,请联系管理员。
转载本站原创文章,请注明出处,并保留原始链接、图片水印。
本站是一个以用户分享为主的开源技术平台,欢迎各类分享!