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上的链接

GitHub

2.FMDB使用指南

  • 下载FMDB框架

  • 在项目中导入第三方框架(导入前需要将sqlite3库文件导入项目)

直接将fmdb文件夹拖入工程,下图箭头指向内容需要勾选:

最后,在使用过程中引入头文件即可。

3.FMDB的三个主要类

  • FMDatabase – 一个FMDatabase对象就表示一个单独的SQLite数据库,用来执行SQLite的命令
  • FMResultSet – 表示FMDatabase执行查询后结果集
  • FMDatabaseQueue – 如果你想在多线程中执行多个查询或更新,你应该使用该类,这样能保证线程安全

4.打开数据库

  • 创建FMDatabase对象,参数为SQLite数据库文件路径。
  1. //获取沙盒路径
  2. NSString *path = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) objectAtIndex:0];
  3. //设置数据库路径
  4. NSString *filePath = [path stringByAppendingPathComponent:@"student.sqlite"];
  5. FMDatabase *database = [FMDatabase databaseWithPath:filePath];
  • 数据库文件路径可以是以下三种之一:

1.具体文件路径。该文件路径如果不存在会自动创建。

2.空字符串(@””)。表示会在临时目录创建一个空的数据库,当FMDatabase 链接关闭时,文件也会被删除。

3.NULL。 将创建一个内存中的临时数据库。同样的,当FMDatabase连接关闭时,数据文件会被销毁。

  • 打开数据库(任何与数据库的交互都必须先打开数据库)
  1. if ([database open]) {
  2. //这里写执行操作代码
  3. } else {
  4. //数据库打开失败
  5. return;
  6. }

检验一下我们的数据库是否创建成功(在沙盒路径中观察数据库是否存在):

通过我们获取到的数据库路径,在Finder中前往文件夹选项输入我们的路径就可以找到我们创建的数据库了。

上一章节我们做了对FMDB的基本介绍,今天我们介绍FMDB的详细使用方法,通过对数据库的增删改查来熟悉FMDB的使用。

1.创建数据表

使用FMDB建立数据表的方法十分简单,先创建sql语句,然后调用executeUpdate方法执行操作:

  1. //创建数据表person(id, name, sex, telephone)
  2. NSString *createSql = [NSString stringWithFormat:@"create table if not exists person (id integer primary key, name text, sex text, telephone text)"];
  3. //执行更新操作(创建表)
  4. if (![database executeUpdate:createSql]) {
  5. NSLog(@"create table failed!");
  6. }

2.添加数据

在executeUpdate方法后直接加sql语句时要注意数据类型的使用,必须使用OC的对象类型

  1. //插入一条记录,(1,jack,male,12345678)
  2. NSString *insertSql = [NSString stringWithFormat:@"insert into person (id, name, sex, telephone) values (%d, ‘%@‘, ‘%@‘, ‘%@‘)", 1, @"jack", @"male", @"12345678"];
  3. //执行更新操作(插入记录)
  4. if (![database executeUpdate:insertSql]) {
  5. NSLog(@"insert failed!");
  6. }
  7. //在executeUpdate后面直接加sql语法时,使用?来表示OC中的对象,integer对应NSNumber,text对应NSString,blob对应NSData,数据内部转换FMDB已经完成,只要sql语法正确就没有问题
  8. if (![database executeUpdate:@"insert into person (id, name, sex, telephone) values (?, ?, ?, ?)", @4, @"gary", @"male", @"99996666"]) {
  9. NSLog(@"insert failed!");
  10. }

3.修改数据

  1. //更新(修改)一条记录,将id = 1的记录姓名修改为mike
  2. NSString *updateSql = [NSString stringWithFormat:@"update person set name = ‘%@‘ where id = 1", @"mike"];
  3. //执行更新操作
  4. if (![database executeUpdate:updateSql]) {
  5. NSLog(@"update failed!");
  6. }

4.删除数据

  1. //删除一条记录,从person表中将id= 2的记录删除
  2. NSString *deleteSql = [NSString stringWithFormat:@"delete from person where id = 2"];
  3. //执行删除操作
  4. if (![database executeUpdate:deleteSql]) {
  5. NSLog(@"delete failed!");
  6. }

5.查询数据

FMDB中一切不是SELECT命令的数据库操作都视为更新,使用executeUpdate方法,SELECT命令的数据库操作使用executeQuery方法。

  1. //查询数据库中记录
  2. NSString *selectSql = [NSString stringWithFormat:@"select * from person"];
  3. //使用executeQuery方法来执行查询语句,使用FMResultSet *来接收查询到的数据
  4. FMResultSet *rs = [database executeQuery:selectSql];
  5. //[rs next]相当于sqlite3_step语句,用来逐行检索数据表中记录
  6. while ([rs next]) {
  7. //使用字段位置查询
  8. NSLog(@"id = %d", [rs intForColumnIndex:0]);
  9. //使用字段名称查询[rs stringForColumn:@"name"]
  10. NSLog(@"name = %@", [rs stringForColumn:@"name"]);
  11. NSLog(@"sex = %@", [rs stringForColumnIndex:2]);
  12. NSLog(@"telephone = %@", [rs stringForColumnIndex:3]);
  13. }

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 对象。

  1. //创建一个FMDatabaseQueue队列
  2. static FMDatabaseQueue *queue;
  1. //初始化队列
  2. queue = [FMDatabaseQueue databaseQueueWithPath:filePath];
  3. //调用inDatabase方法来将需要执行的操作添加到队列queue中去
  4. [queue inDatabase:^(FMDatabase *db) {
  5. //添加需要执行的操作
  6. [db executeUpdate:@"insert into person (id, name, sex, telephone) values (?, ?, ?, ?)", @100, @"test1", @"male", @"11114321"];
  7. [db executeUpdate:@"insert into person (id, name, sex, telephone) values (?, ?, ?, ?)", @101, @"test2", @"male", @"22224321"];
  8. //继续添加想要执行的操作...
  9. }];

使用相似的方法,我们就可以在FMDB中把一些任务包装进事务(在FMDatabaseQueue使用过程中尽量避免嵌套使用,以免造成死锁):

  1. //调用inTransaction方法将需要执行的操作添加到队列中去
  2. [queue inTransaction:^(FMDatabase *db, BOOL *rollback) {
  3. //在事务中添加需要执行的操作,出现异常时及时回滚
  4. if (![db executeUpdate:@"insert into person (id, name, sex, telephone) values (?, ?, ?, ?)", @104, @"test3", @"male", @"11114321"]) {
  5. *rollback = YES;
  6. return ;
  7. }
  8. if (![db executeUpdate:@"insert into person (id, name, sex, telephone) values (?, ?, ?, ?)", @105, @"test4", @"male", @"11114321"]) {
  9. *rollback = YES;
  10. return ;
  11. }
  12. }];

3.总结

FMDatabaseQueue虽然看似一个队列,实际上它本身并不是,它通过内部创建一个Serial的dispatch_queue_t来处理通过inDatabase和inTransaction传入的Blocks,所以当我们在主线程(或者后台)调用inDatabase或者inTransaction时,代码实际上是同步的,这样就避免了多个线程同时访问数据库的问题。如果在后台执行大量的更新操作时,主线程又需要执行少量的数据库操作,那么在后台操作执行完之前,它还是需要等待,这时就会阻塞主线程。

  • 如果在后台使用inDatabase来更新大批量的数据时,可以考虑使用inTransaction,因为后者的更新效率高很多,特别是更新大量操作(如1000条以上)
  • 如果非必须一次性的、完整性的大批量数据,可以考虑使用数据拆解,将大量数据分成较多批少量的数据,再进行更新操作,这样能有效地避免长时间的阻塞
  • 如果UI上不需要在更新数据时产生交互,可以将FMDatabaseQueue放入一个子线程中异步执行,这是一个不错的选择
时间: 2024-11-05 17:21:48

FMDB(一)— 简介的相关文章

FMDB使用简介

转:http://my.oschina.net/youzaiyouzaie/blog/92325 这次要分享的是在iOS中使用SQLite.其实我对SQLite一点概念都没有,因为一直没有机会使用它.有时候突然想到,就会找几本手边的工具书或在网路上找寻在iOS中使用SQLite的教学.但是每次学,每次失败,让我愈来愈讨厌SQLite 囧.直到有一次在网路上发现了FMDB这个东西,才发现不是SQLite很难学,而是SQLite的C API对初学者来说实在太麻烦太琐碎,难度太高,难怪我怎么学都学不会

IOS开发之数据库FMDB

IOS开发之数据库FMDB 1.简介 需求作用:如果需要保存大量的结构较为复杂的数据时候, 使用数据库, 例如交规考试项目 常用的数据库: (1)Microsoft SQL Server 2000/2008:中小企业使用较多 (2)Oracle:比较复杂, 大企业使用较多 (3)Mysql数据库:网站使用较多 (4)sqlite:本地数据库, 访问数据足够快, 直接访问文件  足够简单, 功能相对其他数据库软件不是特别齐全, 足够用了  足够小, 系统不超过1M, 适合在移动端上使用 2. Me

iOS开发 数据库FMDB

iOS开发  数据库FMDB 1.简介 需求作用: 如果需要保存大量的结构较为复杂的数据时候, 使用数据库, 例如交规考试项目 常用的数据库: (1)Microsoft SQL Server 2000/2008, 中小企业使用较多 (2)Oracle 比较复杂, 大企业使用较多 (3)Mysql数据库, 网站使用较多 (4)sqlite: 本地数据库, 访问数据足够快, 直接访问文件 足够简单, 功能相对其他数据库软件不是特别齐全, 足够用了  足够小, 系统不超过1M, 适合在移动端上使用 2

iOS中几种数据持久化方案

概论 所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启后可以继续访问之前保存的数据.在iOS开发中,有很多数据持久化的方案,接下来我将尝试着介绍一下5种方案: plist文件(属性列表) preference(偏好设置) NSKeyedArchiver(归档) SQLite 3 CoreData 沙盒 在介绍各种存储方法之前,有必要说明以下沙盒机制.iOS程序默认情况下只能访问程序自己的目录,这个目录被称为"沙盒". 1.结构 既然沙盒就是一个文件夹,那就看看里面有什么吧

数据储存

概论 所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启后可以继续访问之前保存的数据.在iOS开发中,有很多数据持久化的方案,接下来我将尝试着介绍一下5种方案: plist文件(属性列表) preference(偏好设置) NSKeyedArchiver(归档) SQLite 3 CoreData 沙盒 在介绍各种存储方法之前,有必要说明以下沙盒机制.iOS程序默认情况下只能访问程序自己的目录,这个目录被称为"沙盒". 1.结构 既然沙盒就是一个文件夹,那就看看里面有什么吧

iOS中几种数据持久化方案总结

概论 所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启后可以继续访问之前保存的数据.在iOS开发中,有很多数据持久化的方案,接下来我将尝试着介绍一下5种方案: plist文件(属性列表) preference(偏好设置) NSKeyedArchiver(归档) SQLite 3 CoreData 沙盒 在介绍各种存储方法之前,有必要说明以下沙盒机制.iOS程序默认情况下只能访问程序自己的目录,这个目录被称为"沙盒". 1.结构 既然沙盒就是一个文件夹,那就看看里面有什么吧

数据持久化存储

概论 所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启后可以继续访问之前保存的数据.在iOS开发中,有很多数据持久化的方案,接下来我将尝试着介绍一下5种方案: plist文件(属性列表) preference(偏好设置) NSKeyedArchiver(归档) SQLite 3 CoreData 沙盒 在介绍各种存储方法之前,有必要说明以下沙盒机制.iOS程序默认情况下只能访问程序自己的目录,这个目录被称为“沙盒”. 1.结构 既然沙盒就是一个文件夹,那就看看里面有什么吧.沙盒的目

几种常用的本地话存储方式

概论 所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启后可以继续访问之前保存的数据.在iOS开发中,有很多数据持久化的方案,接下来我将尝试着介绍一下5种方案: plist文件(属性列表) preference(偏好设置) NSKeyedArchiver(归档) SQLite 3 CoreData 沙盒 在介绍各种存储方法之前,有必要说明以下沙盒机制.iOS程序默认情况下只能访问程序自己的目录,这个目录被称为"沙盒". 1.结构 既然沙盒就是一个文件夹,那就看看里面有什么吧

IOS开发——使用数据库

IOS开发——使用FMDB数据库 简介 需求作用: 如果需要保存大量的结构较为复杂的数据的时候,使用数据库,例如交规考试项目 1.数据库的基本介绍 数据库(DB)是一种数据模型组织起来并存放存储管理的数据仓库.它是由文件管理发展起来的,如今的数据库基本上都是关系型数据库. 数据库的基本操作是增.删.查.改. 常见的几种数据库,Oracle,Access,SQL Server,DB2,Mysql 手机上使用的数据库为SQLite,因为它占用资源很好,易于配置等原因. 2.mesaSQLite 使用

iOS中几种数据持久化方案:我要永远地记住你!

作者:@翁呀伟呀 概论 所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启后可以继续访问之前保存的数据.在iOS开发中,有很多数据持久化的方案,接下来我将尝试着介绍一下5种方案: plist文件(属性列表) preference(偏好设置) NSKeyedArchiver(归档) SQLite 3 CoreData 沙盒 在介绍各种存储方法之前,有必要说明以下沙盒机制.iOS程序默认情况下只能访问程序自己的目录,这个目录被称为“沙盒”. 1.结构 既然沙盒就是一个文件夹,那就看看里面