FMDB(一)— 简单介绍
在iOS开发过程中常常会用到数据库方面的操作,但是iOS原生的SQLite API使用起来并不十分友好,对于C语言基础较薄弱的朋友来说。使用起来可能会认为比較不便。于是,一些第三方的对SQLite API进行封装的库就这么出现了。比如本文所要介绍给大家的FMDB。
1.FMDB简单介绍
- 什么是FMDB
FMDB是iOS平台的SQLite数据库框架
FMDB以OC的方式封装了SQLite的C语言API
- 为什么使用FMDB
使用起来更加面向对象,省去了非常多麻烦、冗余的C语言代码
对照苹果自带的Core Data框架,更加轻量级和灵活
提供了多线程安全的数据库操作方法,有效地防止数据混乱
- FMDB在GitHub上的链接
2.FMDB使用指南
- 下载FMDB框架
- 在项目中导入第三方框架(导入前须要将sqlite3库文件导入项目)
直接将fmdb目录拖入project。下图箭头指向内容须要勾选:
最后,在使用过程中引入头文件就可以。
3.FMDB的三个主要类
- FMDatabase – 一个FMDatabase对象就表示一个单独的SQLite数据库,用来运行SQLite的命令
- FMResultSet – 表示FMDatabase运行查询后结果集
- FMDatabaseQueue – 假设你想在多线程中运行多个查询或更新,你应该使用该类,这样能保证线程安全
4.打开数据库
- 创建FMDatabase对象,參数为SQLite数据库文件路径。
- //获取沙盒路径
- NSString *path = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) objectAtIndex:0];
- //设置数据库路径
- NSString *filePath = [path stringByAppendingPathComponent:@"student.sqlite"];
- FMDatabase *database = [FMDatabase databaseWithPath:filePath];
- 数据库文件路径能够是下面三种之中的一个:
表示会在暂时目录创建一个空的数据库。当FMDatabase 链接关闭时,文件也会被删除。
相同的,当FMDatabase连接关闭时。数据文件会被销毁。
- 打开数据库(不论什么与数据库的交互都必须先打开数据库)
- if ([database open]) {
- //这里写运行操作代码
- } else {
- //数据库打开失败
- return;
- }

上一章节我们做了对FMDB的基本介绍。今天我们介绍FMDB的具体用法,通过对数据库的增删改查来熟悉FMDB的使用。
1.创建数据表
使用FMDB建立数据表的方法十分简单。先创建sql语句,然后调用executeUpdate方法运行操作:
- //创建数据表person(id, name, sex, telephone)
- NSString *createSql = [NSString stringWithFormat:@"create table if not exists person (id integer primary key, name text, sex text, telephone text)"];
- //运行更新操作(创建表)
- if (![database executeUpdate:createSql]) {
- NSLog(@"create table failed!");
- }
2.加入数据
在executeUpdate方法后直接加sql语句时要注意数据类型的使用,必须使用OC的对象类型
- //插入一条记录,(1,jack,male,12345678)
- NSString *insertSql = [NSString stringWithFormat:@"insert into person (id, name, sex, telephone) values (%d, '%@', '%@', '%@')", 1, @"jack", @"male", @"12345678"];
- //运行更新操作(插入记录)
- if (![database executeUpdate:insertSql]) {
- NSLog(@"insert failed!");
- }
- //在executeUpdate后面直接加sql语法时。使用?
来表示OC中的对象,integer相应NSNumber。text相应NSString,blob相应NSData。数据内部转换FMDB已经完毕。仅仅要sql语法正确就没有问题
- if (![database executeUpdate:@"insert into person (id, name, sex, telephone) values (?
, ?
, ?, ?)", @4, @"gary", @"male", @"99996666"]) {
- NSLog(@"insert failed!");
- }
3.改动数据
- //更新(改动)一条记录。将id = 1的记录姓名改动为mike
- NSString *updateSql = [NSString stringWithFormat:@"update person set name = '%@' where id = 1", @"mike"];
- //运行更新操作
- if (![database executeUpdate:updateSql]) {
- NSLog(@"update failed!");
- }
4.删除数据
- //删除一条记录,从person表中将id= 2的记录删除
- NSString *deleteSql = [NSString stringWithFormat:@"delete from person where id = 2"];
- //运行删除操作
- if (![database executeUpdate:deleteSql]) {
- NSLog(@"delete failed!");
- }
5.查询数据
FMDB中一切不是SELECT命令的数据库操作都视为更新。使用executeUpdate方法。SELECT命令的数据库操作使用executeQuery方法。
- //查询数据库中记录
- NSString *selectSql = [NSString stringWithFormat:@"select * from person"];
- //使用executeQuery方法来运行查询语句,使用FMResultSet *来接收查询到的数据
- FMResultSet *rs = [database executeQuery:selectSql];
- //[rs next]相当于sqlite3_step语句,用来逐行检索数据表中记录
- while ([rs next]) {
- //使用字段位置查询
- NSLog(@"id = %d", [rs intForColumnIndex:0]);
- //使用字段名称查询[rs stringForColumn:@"name"]
- NSLog(@"name = %@", [rs stringForColumn:@"name"]);
- NSLog(@"sex = %@", [rs stringForColumnIndex:2]);
- NSLog(@"telephone = %@", [rs stringForColumnIndex:3]);
- }
1.多线程下FMDB使用须知
在多线程的环境下。不能多个线程共享一个FMDatabase对象,也不能在多个线程同一时候创建多个FMDatabaseQueue实例来操作同一个数据库。这样可能会造成数据的操作丢失,甚至引起程序的崩溃。由于FMDB是对sqlite的封装,而文件数据库sqlite在同一时刻同意多个进程/线程读。但同一时刻仅仅同意一个线程写。在进行写操作时。数据库文件会被琐定。此时不论什么其它读/写操作都被堵塞,假设堵塞超过5秒钟(默认是5秒,又一次编译sqlite能够改动超时时间)。就报”database is locked”错误。假设线程使用单独的FMDatabase
实例是同意的。但是相同有可能发生database is locked的问题,这是由于多线程对sqlite的竞争引起的。
2.FMDatabaseQueue的使用
使用FMDatabaseQueue能够比較有效的解决多线程下对数据库的訪问。FMDatabaseQueue解决多线程问题的思路大致是:创建一个队列。然后将须要运行的数据库操作放入block中,队列中的block依照加入进队列的顺序依次运行,实际上还是同步的操作,避免了多个线程同一时候对数据库的訪问。
创建一个全局FMDatabaseQueue 对象,这样做的目的是为了避免发生并发訪问数据库的操作。项目开发过程中能够创建一个单例来共享这个FMDatabaseQueue 对象。
- //创建一个FMDatabaseQueue队列
- static FMDatabaseQueue *queue;
- //初始化队列
- queue = [FMDatabaseQueue databaseQueueWithPath:filePath];
- //调用inDatabase方法来将须要运行的操作加入到队列queue中去
- [queue inDatabase:^(FMDatabase *db) {
- //加入须要运行的操作
- [db executeUpdate:@"insert into person (id, name, sex, telephone) values (?
, ?, ?, ?)", @100, @"test1", @"male", @"11114321"];
- [db executeUpdate:@"insert into person (id, name, sex, telephone) values (?
, ?, ?
, ?)", @101, @"test2", @"male", @"22224321"];
- //继续加入想要运行的操作...
- }];
使用类似的方法,我们就能够在FMDB中把一些任务包装进事务(在FMDatabaseQueue使用过程中尽量避免嵌套使用,以免造成死锁):
- //调用inTransaction方法将须要运行的操作加入到队列中去
- [queue inTransaction:^(FMDatabase *db, BOOL *rollback) {
- //在事务中加入须要运行的操作,出现异常时及时回滚
- if (![db executeUpdate:@"insert into person (id, name, sex, telephone) values (?, ?, ?, ?)", @104, @"test3", @"male", @"11114321"]) {
- *rollback = YES;
- return ;
- }
- if (![db executeUpdate:@"insert into person (id, name, sex, telephone) values (?, ?, ?, ?)", @105, @"test4", @"male", @"11114321"]) {
- *rollback = YES;
- return ;
- }
- }];
3.总结
FMDatabaseQueue尽管看似一个队列,实际上它本身并非。它通过内部创建一个Serial的dispatch_queue_t来处理通过inDatabase和inTransaction传入的Blocks,所以当我们在主线程(或者后台)调用inDatabase或者inTransaction时,代码实际上是同步的,这样就避免了多个线程同一时候訪问数据库的问题。假设在后台运行大量的更新操作时。主线程又须要运行少量的数据库操作,那么在后台操作运行完之前,它还是须要等待。这时就会堵塞主线程。
- 假设在后台使用inDatabase来更新大批量的数据时,能够考虑使用inTransaction。由于后者的更新效率高非常多,特别是更新大量操作(如1000条以上)
- 假设非必须一次性的、完整性的大批量数据,能够考虑使用数据拆解。将大量数据分成较多批少量的数据。再进行更新操作,这样能有效地避免长时间的堵塞
- 假设UI上不须要在更新数据时产生交互,能够将FMDatabaseQueue放入一个子线程中异步运行,这是一个不错的选择
FMDB(一)— 简单介绍的更多相关文章
- iOS开发数据库篇—FMDB简单介绍
iOS开发数据库篇—FMDB简单介绍 一.简单说明 1.什么是FMDB FMDB是iOS平台的SQLite数据库框架 FMDB以OC的方式封装了SQLite的C语言API 2.FMDB的优点 使用起来 ...
- iOS 数据库第三方FMDB的简单使用
一 FMDB的简单说明及介绍 FMDB的github地址 https://github.com/ccgus/fmdb FMDB是一款简洁的,易用的封装库,简单介绍一下FMDB的使用 在FMDB下载文件 ...
- [原创]关于mybatis中一级缓存和二级缓存的简单介绍
关于mybatis中一级缓存和二级缓存的简单介绍 mybatis的一级缓存: MyBatis会在表示会话的SqlSession对象中建立一个简单的缓存,将每次查询到的结果结果缓存起来,当下次查询的时候 ...
- 利用Python进行数据分析(7) pandas基础: Series和DataFrame的简单介绍
一.pandas 是什么 pandas 是基于 NumPy 的一个 Python 数据分析包,主要目的是为了数据分析.它提供了大量高级的数据结构和对数据处理的方法. pandas 有两个主要的数据结构 ...
- 利用Python进行数据分析(4) NumPy基础: ndarray简单介绍
一.NumPy 是什么 NumPy 是 Python 科学计算的基础包,它专为进行严格的数字处理而产生.在之前的随笔里已有更加详细的介绍,这里不再赘述. 利用 Python 进行数据分析(一)简单介绍 ...
- yii2的权限管理系统RBAC简单介绍
这里有几个概念 权限: 指用户是否可以执行哪些操作,如:编辑.发布.查看回帖 角色 比如:VIP用户组, 高级会员组,中级会员组,初级会员组 VIP用户组:发帖.回帖.删帖.浏览权限 高级会员组:发帖 ...
- angular1.x的简单介绍(二)
首先还是要强调一下DI,DI(Denpendency Injection)伸手获得,主要解决模块间的耦合关系.那么模块是又什么组成的呢?在我看来,模块的最小单位是类,多个类的组合就是模块.关于在根模块 ...
- Linux的简单介绍和常用命令的介绍
Linux的简单介绍和常用命令的介绍 本说明以Ubuntu系统为例 Ubuntu系统的安装自行百度,或者参考http://www.cnblogs.com/CoderJYF/p/6091068.html ...
- iOS-iOS开发简单介绍
概览 终于到了真正接触IOS应用程序的时刻了,之前我们花了很多时间去讨论C语言.ObjC等知识,对于很多朋友而言开发IOS第一天就想直接看到成果,看到可以运行的IOS程序.但是这里我想强调一下,前面的 ...
随机推荐
- 12、Django实战第12天:课程机构列表页数据展示
今天完成的是课程机构列表页.... 1.启动服务,进入xadmin后,添加5个城市信息用作测试数据 2.添加课程机构,其中有一项要上传封面图的地方要注意 封面图上传路径是我们在models中设置好的 ...
- luogu P1136 迎接仪式
luogu P1136 迎接仪式 本题的难点是状态设计, n^2*m 的状态设计转移太过垄杂,emmmm反正我写不出来QAQ 参考了题解 /*相同字符不用调换,一个字符最多被调换一次否则会有等价多方案 ...
- [BZOJ2111][ZJOI2010]Perm排列计数(组合数学)
题意就是求一个n个点的堆的合法形态数. 显然,给定堆中所有数的集合,则这个堆的根是确定的,而由于堆是完全二叉树,所以每个点左右子树的大小也是确定的. 设以i为根的堆的形态数为F(i),所以F(i)+= ...
- 【kd-tree】bzoj1176 [Balkan2007]Mokia
裸题不多说,注意在sqrt(n*log(n))次插入后重构树以保持深度. #include<cstdio> #include<cmath> #include<algori ...
- 【尺取法】【Multiset】bzoj1342 [Baltic2007]Sound静音问题
O(n)地枚举所有长度为k的段,每次暴力转移. 转移的时候只是从最后插入一个数,从前面删去一个数. 计算的时候要取当前的max和min. 用multiset(∵元素是可重的)以上这些操作都是O(log ...
- SpringMVC实现操作的第二种方式
一: 运行效果: 点击提交之后显示效果 二: (1).web.xml <?xml version="1.0" encoding="UTF-8"?> ...
- [转]C++中关于new和delete的使用
转载的地址 近一直在啃 C++ Primer 中文版第4版,发现 C++中new和delete应用遍布全书,现对其使用作简单总结.在C++中,可以使用new和delete动态创建和释放数组或者单个对象 ...
- TDiocpCoderTcpServer返回数据记录有条数限制的问题
TDiocpCoderTcpServer返回数据记录有条数限制的问题 在使用TDiocpCoderTcpServer控件返回查询数据的时候,发现当记录条数超过一定数量的时候(比方有人反试图返回30万条 ...
- 使用ab.exe监测100个并发/100次请求情况下同步/异步访问数据库的性能差异
ab.exe介绍 ab.exe是apache server的一个组件,用于监测并发请求,并显示监测数据 具体使用及下载地址请参考:http://www.cnblogs.com/gossip/p/439 ...
- Thinkphp2.1爆出重大安全漏洞
thinkphp 2.1的版本 我们来分析下漏洞吧 官方发布了一个安全补丁 表述是:该URL安全漏洞会造成用户在客户端伪造URL,执行非法代码. 官方的补丁: /trunk/ThinkPHP/Lib/ ...