对FMDB的封装JRDB
在自己开发中,每次用到数据库都会纠结是使用CoreData还是FMDB。CoreData虽然Api简单,但是调用栈非常复杂,要初始化一个Context需要至少20行代码。显然,对于这种这么恶心的情况,我们的大Github必须有人会跳出来解决这种问题。于是就出现了MagicRecord这个对CoreData的封装库。一开始遇到这个库的时候,好用到几乎让我想把所有项目的数据库都换成CoreData了。两句话解决CoreData调用栈的初始化,一句话完成数据库版本升级和自动数据合并更新(虽然我们很少用到)。然而这并不能解决一个根本性的问题,CoreData中的每个Object都要和一个context进行绑定,导致我们很多业务需求需要创建自己的私有context,然后再需要更新的时候保存到主context中。这又导致了我们在controller中或者在自己的业务类中维护多一个私有context属性。所以,最后还是选择了FMDB进行封装。之前自己搞过Java后台,将FMDB进行Hibernate式的封装,使用runtime解析,不用继承任何基类(swift中要继承NSObject),只要实现一个持久化协议并实现方法即可,屏蔽基本的数据库和表操作。
项目简介:
JRDB:一个对FMDB进行类Hibernate封装的iOS库,支持Objective-C 和 Swift。
Description
使用分类的模式,模仿Hibernate,对FMDB进行简易封装
支持pod 安装 『pod 'JRDB'』,Podfile需要添加 use_framework!
使用协议,不用继承基类,对任意NSObject可以进行入库操作
支持swift 和 Objective-C
支持数据类型:基本数据类型(int,double,等),String,NSData,NSNumber,NSDate
注:Swift的基本数据类型,不支持Option类型,既不支持Int?Int!等,对象类型支持Option类型
Installation(安装)
|
1
2
3
|
use_frameworks!pod 'JRDB'@import JRDB; |
Usage
Save(保存)
Objective-C
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
Person *p = [[Person alloc] init];p.a_int = 1;p.b_unsigned_int = 2;p.c_long = 3;p.d_long_long = 4;p.e_unsigned_long = 5;p.f_unsigned_long_long = 6;p.g_float = 7.0;p.h_double = 8.0;p.i_string = @"9";p.j_number = @10;p.k_data = [NSData data];p.l_date = [NSDate date];[p jr_save]; |
Swift
Swift中需要入库的类需要继承NSObject(使用到runtime)
The Object that you want to persistent should inherit from NSObject
|
1
2
3
4
5
|
let p = Person()p.name = "name"p.age = 10p.birthday = NSDate()p.jr_save() |
Update(更新)
|
1
2
3
|
Person *p = [Person jr_findAll].firstObject;p.name = @"abc";[p jr_update columns:nil]; |
column: 需要更新的字段名,传入空为全量更新
Delete(删除)
|
1
2
|
Person *p = [Person jr_findAll].firstObject;[p jr_delete]; |
Select(查找)
常规查找
|
1
2
3
|
Person *p = [Person jr_findByPrimaryKey:@"111"];NSArray *list = [Person jr_findAll];NSArray *list1 = [Person jr_findAllOrderBy:@"_age" isDesc:YES]; |
条件查询
|
1
2
3
4
5
6
7
8
|
NSArray *condis = @[ [JRQueryCondition condition:@"_l_date < ?" args:@[[NSDate date]] type:JRQueryConditionTypeAnd], [JRQueryCondition condition:@"_a_int > ?" args:@[@9] type:JRQueryConditionTypeAnd],];NSArray *arr = [Person jr_findByConditions:condis groupBy:@"_room" orderBy:@"_age" limit:@" limit 0,13 " isDesc:YES]; |
SQL
|
1
2
|
NSString *sql = @"select * from Person where age = ?";NSArray *list = [Person jr_executeSql:sql args:@[@10]]; |
Other(其他)
协议:JRPersistent
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
@protocol JRPersistent @required- (void)setID:(NSString * _Nullable)ID;- (NSString * _Nullable)ID;@optional/** * 返回不用入库的对象字段数组 * The full property names that you want to ignore for persistent * @return array */+ (NSArray * _Nullable)jr_excludePropertyNames;/** * 返回自定义主键字段 * @return 字段全名 */+ (NSString * _Nullable)jr_customPrimarykey;/** * 返回自定义主键值 * @return 主键值 */- (id _Nullable)jr_customPrimarykeyValue;@end |
主键
默认每个Object的主键为ID, UUID字符串。
可以实现 jr_customPrimarykey 以及 jr_customPrimarykeyValue 方法,自定义主键。
默认NSObject分类实现
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
@interface NSObject (JRDB) (...methods)@endJRDBMgr@interface JRDBMgr : NSObject@property (nonatomic, strong) FMDatabase *defaultDB;+ (instancetype)shareInstance;+ (FMDatabase *)defaultDB;- (FMDatabase *)createDBWithPath:(NSString *)path;- (void)deleteDBWithPath:(NSString *)path;/** * 在这里注册的类,使用本框架的数据库将全部建有这些表 * @param clazz 类名 */- (void)registerClazzForUpdateTable:(Class)clazz;- (NSArray *)registedClazz;/** * 更新默认数据库的表(或者新建没有的表) * 更新的表需要在本类先注册 */- (void)updateDefaultDB;- (void)updateDB:(FMDatabase *)db;@end |
JRDBMgr持有一个默认数据库(~/Documents/jrdb/jrdb.sqlite),任何不指定数据库的操作,都在此数据库进行操作。默认数据库可以自行设置。
Method
|
1
|
- (void)registerClazzForUpdateTable:(Class)clazz; |
在JRDBMgr中注册的类,可以使用
|
1
|
-(void)updateDB:(FMDatabase *)db |
进行统一更新或者创建表。
Table Operation(表操作)
Create(建表)
|
1
2
3
4
5
6
7
8
9
10
11
|
// FMDatabase+JRDB 方法[[JRDBMgr defaultDB] createTable4Clazz:[Person class]];[Person jr_createTable];// 删除原有的表,重新创建[[JRDBMgr defaultDB] truncateTable4Clazz:[Person class]];[Person jr_truncateTable];//保存时,若发现没有表,将自动创建[person jr_save];Update 【更新表】[[JRDBMgr defaultDB] updateTable4Clazz:[Person class]];[Person jr_updateTable]; |
更新表时,只会添加不存在的字段,不会修改字段属性,不会删除字段,若有需要,需要自行写sql语句进行修改
Drop(删表)
|
1
2
|
[[JRDBMgr defaultDB] dropTable4Clazz:[Person class]];[Person jr_dropTable]; |
Thread Operation(线程操作)
多线程操作使用FMDB自带的 FMDatabaseQueue
|
1
2
3
|
[person jr_saveWithComplete:^(BOOL success) { NSLog(@"%d", success);}]; |
任何带complete block的操作,都将放入到FMDatabaseQueue进行顺序执行
注:所有需要立刻返回结果,或者影响其他操作的数据库操作,都建议放在主线程进行更新,大批量更新以及多线程操作数据库时,请使用带complete block的操作。
MoreUsage
查看FMDatabase+JRDB.h
项目地址:http://download.csdn.net/detail/hbblzjy/9557232
对FMDB的封装JRDB的更多相关文章
- FMDB简单封装和使用
工具:火狐浏览器+SQLite Manager插件 ; Xcode; FMDB库; 效果: 项目地址: https://github.com/sven713/PackFMDB 主要参考这两篇博客: 1 ...
- 一个使用方便的对FMDB进行封装的框架和一个可以切应用图标的应用
框架的git地址 :https://github.com/Joker-King/JKDBModel 切割图标的应用地址 http://pan.baidu.com/s/1kVjflwr
- 第六十二篇、AFN3.0封装网络请求框架,支持缓存
1.网络请求 第一种实现方式: 功能:GET POST 请求 缓存逻辑: 1.是否要刷新本地缓存,不需要就直接发起无缓存的网络请求,否则直接读取本地数据 2.需要刷新本地缓存,先读取本地数据,有就返回 ...
- iOS数据持久化之数据库:SQLite和FMDB
SQLite: SQLite是一款轻量级型的数据库,资源占用少.性能良好和零管理成本,具有零配置(无需安装和管理配置).独立(没有额外依赖).储存在单一磁盘文件中的一个完整的数据库.源码完全的开源.比 ...
- iOS之数据持久化方案
概论 所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启后可以继续访问之前保存的数据.在iOS开发中,有很多数据持久化的方案,接下来我将尝试着介绍一下5种方案: plist文件(属性列表) ...
- iOS的数据持久化
所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启后可以继续访问之前保存的数据.在iOS开发中,有很多数据持久化的方案,接下来我将尝试着介绍一下5种方案: plist文件(属性列表) pr ...
- IOS开发涉及有点概念&相关知识点
前言,IOS是基于UNIX的,用C/C+/OC直通系统底层,不想android有个jvm. 首先还是系统架构的分层架构 1.核心操作系统层 Core OS,就是内存管理.文件系统.电源管理等 2.核心 ...
- iOS数据持久化
在iOS中,实现数据持久化一般分为4大种: 1.属性列表 2.对象归档 3.SQLite 4.Core Data 一.属性列表 NSUserDefaults类的使用和NSKeyedArchiver有很 ...
- 一些Iphone sqlite 的包装类
相信很多人用iphone的Sqlite不会直接用C的方法,要么自己包装一层Object c的访问方法,要么用CoreData,下面我整理些目前所了结的一些Sqlite 包装类. 1.CoreData ...
随机推荐
- cocos2dx 3.2之Lua打飞机项目
1 创建lua打飞机项目 cocos new T32Lua -dE:\Installed\cocos2d-x-3.2\cocos2d-x-3.2\projects -l lua 2 ...
- 1086. Tree Traversals Again (25)
题目如下: An inorder binary tree traversal can be implemented in a non-recursive way with a stack. For e ...
- Dynamics Crm 2011 Or 2013 IFD 部署一段时间后,CA验证问题
以下错误描述摘自博客:http://blog.csdn.net/qzw4549689/article/details/14451257 IFD部署一段时间后,大概一年,突然出现从IFD登录页面登录后, ...
- Android必知必会-Fragment监听返回键事件
如果移动端访问不佳,请尝试 Github版<–点击左侧 背景 项目要求用户注册成功后进入修改个人资料的页面,且不允许返回到上一个页面,资料修改完成后结束当前页面,进入APP主页. 由于是使用多个 ...
- Android简易实战教程--第十五话《在外部存储中读写文件》
第七话里面介绍了在内部存储读写文件 点击打开链接. 这样有一个比较打的问题,假设系统内存不够用,杀本应用无法执行,或者本应用被用户卸载重新安装后.以前保存的用户名和密码都不会得到回显.所以,有必要注意 ...
- 发现----Android Demo
时间悄悄的走,转眼来实习已经三个月了,三个月的时间,小编慢慢的成长着,从刚开始的电商项目到现在的车段子项目,小编在走过一个又一个项目的同时,走过了一个又一个战胜自己的奇迹,每次遇到一个新的技术点,小编 ...
- java操作properties配置文件
Java中有个类Properties(Java.util.Properties),主要用于读取Java的配置文件,将一些可能需要变化的值存放在properties中进行配置,通常为为.properti ...
- linux 定时任务详解 按秒设定
实现linux定时任务有:cron.anacron.at等,这里主要介绍cron服务. 名词解释: cron是服务名称,crond是后台进程,crontab则是定制好的计划任务表. 软件包安装: 要使 ...
- printk的用法
printk的用法 内核通过 printk() 输出的信息具有日志级别,日志级别是通过在 printk() 输出的字符串前加一个带尖括号的整数来控制的,如 printk("<6> ...
- JQuery EasyUI combobox(下拉列表框)
下拉列表框 继承 $.fn.combo.defaults. 重写 $.fn.combobox.defaults. 组合框显示一个可编辑的文本框和下拉列表,用户选择一个或多个值.用户可以直接输入文 ...