使用FMDB多线程访问数据库,及database is locked的问题

简介:

目录(?)[+]

每日更新关注:http://weibo.com/hanjunqiang  新浪微博

今天终于解决了多线程同时访问数据库时,报数据库锁定的问题,错误信息是:

Unknown error finalizing or resetting statement (5: database is locked)

最后通过FMDatabaseQueue解决了这个问题,本文总结一下:

FMDatabase不能多线程使用同一个实例

多线程访问数据库,不能使用同一个FMDatabase的实例,否则会发生异常。如果线程使用单独的FMDatabase实例是允许的,但是同样有可能发生database is locked的问题。这是由于多线程对sqlite的竞争引起的

我的app一开始就是多线程使用单独的FMDatabase实例访问数据库,虽然没有引起crash,但是还是出现了database is locked问题,造成很多数据没有如预期写入数据库

使用FMDatabaseQueue,问题依旧

后来上FMDB的官网看了文档,确认用FMDatabaseQueue可以解决这个问题,API也比较简单:

[objc]  view plain  copy
  1. NSString *dbFilePath = [PathResolver databaseFilePath];  
  2. queue = [FMDatabaseQueue databaseQueueWithPath:dbFilePath];  
  3. [queue inDatabase:^(FMDatabase *db){  
  4.     // access db  
  5. }];  
每日更新关注 : http://weibo.com/hanjunqiang   新浪微博


但是实际测试了一下,还是database is locked

读了一下相关的源码,FMDatabaseQueue解决这个问题的思路是:创建一个队列,然后将放入队列的block顺序执行,这样避免了多线程同时访问数据库

而我的代码是多线程各创建FMDatabaseQueue的实例,所以其实有多个队列,因此还是存在数据库竞争的问题,和用FMDatabase时是一样的

共享同一个FMDatabaseQueue实例

于是接下来我让每个线程使用同一个Queue实例,问题就顺利解决了

实现的方式,一开始我想给FMDatabase增加一个单例方法,但是这样以后升级FMDB会比较麻烦,所以最后我是创建了一个Helper类

[objc]  view plain  copy
  1. @implementation LosDatabaseHelper  
  2.   
  3. {  
  4.     FMDatabaseQueue* queue;  
  5. }  
  6.   
  7. -(id) init  
  8. {  
  9.     self = [super init];  
  10.     if(self){  
  11.         NSString *dbFilePath = [PathResolver databaseFilePath];  
  12.         queue = [FMDatabaseQueue databaseQueueWithPath:dbFilePath];  
  13.     }  
  14.     return self;  
  15. }  
  16.   
  17. +(LosDatabaseHelper*) sharedInstance  
  18. {  
  19.     static dispatch_once_t pred = 0;  
  20.     __strong static id _sharedObject = nil;  
  21.     dispatch_once(&pred, ^{  
  22.         _sharedObject = [[self alloc] init];  
  23.     });  
  24.     return _sharedObject;  
  25. }  
  26.   
  27. -(void) inDatabase:(void(^)(FMDatabase*))block  
  28. {  
  29.     [queue inDatabase:^(FMDatabase *db){  
  30.         block(db);  
  31.     }];  
  32. }  
  33.   
  34. @end  

每日更新关注:http://weibo.com/hanjunqiang  新浪微博

系统中其他的类,使用这个Helper类的单例,这样保证了全局只有唯一的FMDatabaseQueue实例。注意,因为Helper内部持有的是 FMDatabaseQueue,所以可以这么做,如果包装的是FMDatabase类,就绝对会有问题。因为FMDatabase实例不能在多线程环境 共享

使用FMDatabaseQueue之后,管理db

原本使用FMDatabase类,需要手工调用db的open和close方法

但是用FMDatabaseQueue,不需要调用open,因为查看代码发现,Queue已经open了。至于要不要close,我也不确定,因 为官方的sample code没有调用close。实际应用中,我也没有调用,好像没有问题。如果需要close的话,我想可以在Helper类的公共方法里增加调用 close queue就可以了。下面是close的源码:

[objc]  view plain  copy
  1. - (void)close {  
  2.     FMDBRetain(self);  
  3.     dispatch_sync(_queue, ^() {  
  4.         [_db close];  
  5.         FMDBRelease(_db);  
  6.         _db = 0x00;  
  7.    });  
  8.    FMDBRelease(self);  
  9. }  

所以,使用Queue,是不需要自己打开和关闭db的。但是如果使用了FMResultSet,rs倒是需要关闭,否则会报warning:
[objc]  view plain  copy
  1. if ([db hasOpenResultSets]) {  
  2.     NSLog(@"Warning: there is at least one open result set around after performing [FMDatabaseQueue inDatabase:]");  

为了不看到warning,我都在block里调用了[rs close]

刷新数据库文件路径

具体到我们的应用,还有一个特殊问题需要考虑。因为我们的APP可以切换账户,而账户的db文件是独立的。所以当用户重新登录的时候,需要刷新一下Helper的queue

[objc]  view plain  copy
  1. +(void) refreshDatabaseFile  
  2. {  
  3.     LosDatabaseHelper *instance = [self sharedInstance];  
  4.     [instance doRefresh];  
  5. }  
  6.   
  7. -(void) doRefresh  
  8. {  
  9.     NSString *dbFilePath = [PathResolver databaseFilePath];  
  10.     queue = [FMDatabaseQueue databaseQueueWithPath:dbFilePath];  
  11. }  

如果不这么做,由于Helper是单例,那么切换账户以后,用户B访问的还是用户A的数据库。刷新的调用,一般放在登录之后,进入主页面之前就可以了

队列和线程

在debug过程中,顺便看到一个现象。虽然多个block都是放到同一个队列里,但是其实是跑在不同的thread里


不要混淆队列和线程的概念,使用GCD时,开发者关注的是把block放到队列中,但是同一个队列其实可以对应多个thread,为block分配thread,是GCD框架负责的,开发者不需要关注。只要把操作放到合适的队列里,GCD就会完成线程的创建,分配与回收

每日更新关注:http://weibo.com/hanjunqiang  新浪微博




原文地址: http://blog.csdn.net/qq_31810357/article/details/50550247
相关文章
|
1月前
|
数据采集 Java API
Jsoup库能处理多线程下载吗?
Jsoup库能处理多线程下载吗?
|
8月前
|
并行计算 安全 程序员
【C++】—— C++11之线程库
【C++】—— C++11之线程库
104 0
|
8月前
|
Java 调度 Python
深入解析 Python asyncio 库:如何使用线程池实现高效异步编程
深入解析 Python asyncio 库:如何使用线程池实现高效异步编程
510 0
|
4月前
|
数据库 数据库管理
qt对sqlite数据库多线程的操作
本文总结了在Qt中进行SQLite数据库多线程操作时应注意的四个关键问题,包括数据库驱动加载、加锁、数据库的打开与关闭,以及QsqlQuery变量的使用。
247 1
|
3月前
|
NoSQL 关系型数据库 MySQL
AWS Database Migration Service 助力数据库搬迁
AWS Database Migration Service 助力数据库搬迁
|
5月前
|
安全 Java
Java模拟生产者-消费者问题。生产者不断的往仓库中存放产品,消费者从仓库中消费产品。其中生产者和消费者都可以有若干个。在这里,生产者是一个线程,消费者是一个线程。仓库容量有限,只有库满时生产者不能存
该博客文章通过Java代码示例演示了生产者-消费者问题,其中生产者在仓库未满时生产产品,消费者在仓库有产品时消费产品,通过同步机制确保多线程环境下的线程安全和有效通信。
|
6月前
|
SQL 关系型数据库 MySQL
云服务器 ECS产品使用问题之出现“1044 - Access denied for user ‘root‘@‘%‘ to database ‘数据库名称‘”这样的错误,该怎么办
云服务器ECS(Elastic Compute Service)是各大云服务商阿里云提供的一种基础云计算服务,它允许用户租用云端计算资源来部署和运行各种应用程序。以下是一个关于如何使用ECS产品的综合指南。
|
5月前
|
数据采集 存储 监控
如何使用pholcus库进行多线程网页标题抓取以提高效率?
如何使用pholcus库进行多线程网页标题抓取以提高效率?
|
6月前
|
网络协议 安全 Python
我们将使用Python的内置库`http.server`来创建一个简单的Web服务器。虽然这个示例相对简单,但我们可以围绕它展开许多讨论,包括HTTP协议、网络编程、异常处理、多线程等。
我们将使用Python的内置库`http.server`来创建一个简单的Web服务器。虽然这个示例相对简单,但我们可以围绕它展开许多讨论,包括HTTP协议、网络编程、异常处理、多线程等。
|
6月前
|
数据库
数据库bug-[08001] Could not create connection to database server. Attempted reconnect 3,主机名ip必须写对
数据库bug-[08001] Could not create connection to database server. Attempted reconnect 3,主机名ip必须写对