sqlite: Error Code : 5 (SQLITE_BUSY) (database is locked (code 5): , while compiling: PRAGMA journal_mode)
今天遇到了一个很奇怪的问题,登录完成后,程序会莫名crash, 报了下面的错误:
sqlite: Error Code : (SQLITE_BUSY) (database is locked (code ): , while compiling: PRAGMA journal_mode)
经过分析发现是数据库被locked,从而导致在调用
mDatabase = mDBHelper.getWritableDatabase();
时发生了crash, 而android上面同一时间向sqlite里写数据只允许有一个sqlite connection,所以发生了crash, 不过问题根源还是在数据库locked, 那么如何发生locked的呢?
仔细检查后发现:
private synchronized void copyTableFromAnonymousTable() {
SQLiteDatabase db = openDatabase();
try {
if (db.isOpen()) {
db.execSQL("create table if not exists "+ FLIGHT_LOG_TABLE_NAME +
"(id integer primary key autoincrement, " +
""+DBConfig.FLYING_LOG_USER_ID+" integer, "+
""+DBConfig.FLYING_LOG_USER_NAME+" varchar(100), "+
""+DBConfig.FLYING_LOG_HOME_LAT+" double, "+
""+DBConfig.FLYING_LOG_HOME_LON+" double, "+
""+DBConfig.FLYING_LOG_LAST_LAT+" double, "+
""+DBConfig.FLYING_LOG_LAST_LON+" double, "+
""+DBConfig.FLYING_LOG_LOCATION+" varchar(250), "+
""+DBConfig.FLYING_LOG_CITY+" varchar(50), "+
""+DBConfig.FLYING_LOG_DRONE_TYPE+" integer, "+
""+DBConfig.FLYING_LOG_DRONE_TYPE_STR+" varchar(100), "+
""+DBConfig.FLYING_LOG_DRONE_SSID+" varchar(150), "+
""+DBConfig.FLYING_LOG_DRONE_SN+" varchar(150), "+
""+DBConfig.FLYING_LOG_TAKEOFF_TIME+" long, "+
""+DBConfig.FLYING_LOG_LANDING_TIME+" long, "+
""+DBConfig.FLYING_LOG_DURATION+" long, "+
""+DBConfig.FLYING_LOG_TAKEOFF_TIME_STR+" varchar(150), "+
""+DBConfig.FLYING_LOG_RAW_OFFSET+" long, "+
""+DBConfig.FLYING_LOG_APP_VERSION+" varchar(100), "+
""+DBConfig.FLYING_LOG_FC_VERSION+" varchar(100), "+
""+DBConfig.FLYING_LOG_RC_VERSION+" varchar(100), "+
""+DBConfig.FLYING_LOG_CAM_VERSION+" varchar(100), "+
""+DBConfig.FLYING_LOG_MAX_HEIGHT+" double, "+
""+DBConfig.FLYING_LOG_MAX_SPEED+" double, "+
""+DBConfig.FLYING_LOG_MAX_DISTANCE +" double, "+
""+DBConfig.FLYING_LOG_TOTAL_MILEAGE+" double, "+
""+DBConfig.FLYING_LOG_FLIGHT_STATUS+" double, "+
""+DBConfig.FLYING_LOG_PLATFORM+" varchar(50), "+
""+DBConfig.FLYING_LOG_EXTEND_DATA1+" varchar(100), "+
""+DBConfig.FLYING_LOG_EXTEND_DATA2+" varchar(100), "+
""+DBConfig.FLYING_LOG_EXTEND_DATA3+" varchar(100), "+
""+DBConfig.FLYING_LOG_EXTEND_DATA4+" varchar(100)"
+")");
db.beginTransaction();
db.execSQL("INSERT INTO "+FLIGHT_LOG_TABLE_NAME+" SELECT * FROM "+FLIGHT_DEFAULT_TABLE_NAME+"");
db.setTransactionSuccessful(); db.endTransaction();
}
} catch (Exception e) {
e.printStackTrace();
} finally {
closeDatabase();
}
db.execSQL("INSERT INTO "+FLIGHT_LOG_TABLE_NAME+" SELECT * FROM "+FLIGHT_DEFAULT_TABLE_NAME+"");
执行上述SQL语句时是有问题的,而这个问题被try catch 捕获掉了,但是:
db.endTransaction();
却没有被执行!后来将 db.endTransaction(); 移至finally 代码块中发现程序变好了。
以前一直觉得只要记得closeDatabase就好了,却没注意到不 end transaction 也会造成问题,在此记录一下,引以为诫。
关于如何unlock的问题
其实我发现数据库被Locked了之后的第一反应时是如何去unlock。不过很遗憾,我没有在现有的android API里找到类似于数据库lock/unlock的方法。不过android 倒是提供了一个
isDbLockedByCurrentThread() 这样的方法用于检测当前数据库是否有被lock, 网上很多的数据库打开方式都是类似于下列代码:
private synchronized SQLiteDatabase openDatabase() {
if (mOpenCounter.incrementAndGet() == 1 || mDatabase == null) {
mDatabase = mDBHelper.getWritableDatabase();
}
return mDatabase;
}
这种方式虽然貌似解决了多个 数据库connection的问题,但是个人以为还是不太完善,像我遇到的这种情况,最终crash就发生在
mDatabase = mDBHelper.getWritableDatabase();
这行代码里。因此个人以为保险起见,可以在上述代码之前添加一个诸如下列的代码:
private synchronized SQLiteDatabase openDatabase() {
if (mOpenCounter.incrementAndGet() == 1 || mDatabase == null) {
if (mDatabase == null) {
mDatabase = mDBHelper.getWritableDatabase();
}
else {
while (mDatabase.isDbLockedByCurrentThread()) {
try {
Thread.sleep(100);
}
catch (InterruptedException e) {
e.printStackTrace();
}
}
mDatabase = mDBHelper.getWritableDatabase();
}
}
return mDatabase;
}
可能会更好点,个人看法。
sqlite: Error Code : 5 (SQLITE_BUSY) (database is locked (code 5): , while compiling: PRAGMA journal_mode)的更多相关文章
- sqlite:多线程操作数据库“database is locked”解决方法(二)
上一篇博客<sqlite:多线程操作数据库“database is locked”解决方法>通过注册延时函数的方法来处理数据库被锁的问题.此方法固然能解决问题,但是在多个线程向数据库写入大 ...
- sqlite:多线程操作数据库“database is locked”解决方法
1. 使sqlite支持多线程(不确定是否非加不可,暂且加上,以备后患) 可以在编译时/启动时/运行时选择线程模式,参考:http://www.cnblogs.com/liaj/p/4015219.h ...
- sqlite遇到database is locked问题的完美解决
这两天在项目中用大强度大频率的方法测试时遇到sqlite报database is locked的问题,分析下来原因是sqlite对数据库做修改操作时会做(文件)锁使得其它进程同一时间使用时会报该错误( ...
- SQLITE报错database is locked的解决办法
用firedac连接SQLITE数据库,空间tdbedit绑定字段name,如下语句修改其值时报错. procedure TForm1.Button3Click(Sender: TObject);be ...
- 解决SQLite database is locked
前些时候,同事在站点服务端使用SQlite存储一些临时数据,但是在多人并发的时候Sqlite会抛出异常:The database file is locked , database is locked ...
- sqlite 报错:database is locked
在sqlite批量添加数据时,报错:database is locked. 解决办法:将db路径由相对路径设置为绝对路径.
- dpkg: error: dpkg status database is locked by another process 解决方法
使用dpkg -i/apt命令安装,报错: ------------------------------------------------------------- dpkg: error: dpk ...
- 解决SQLite中的 database is locked
前些时候,同事在站点服务端使用SQlite存储一些临时数据,但是在多人并发的时候Sqlite会抛出异常:The database file is locked , database is locked ...
- SQLITE 多进程查询出错database is locked
程序比较简单: 父进程查询数据库A表,没有更新操作 子进程同时查询数据库A表,查询出来的内容更新B表. 两个进程都放到while(1)循环中,速度慢的话就是2S执行一次就没有错,执行的速度快的话就会报 ...
随机推荐
- 转 PHP5+APACHE2.2配置
初学php,配置起来老出问题,找了篇不错的帖子,一试就通过了,所以就顺带着转了过来. 不过在我安装phpMyAdmin的时候还是发现这篇文章的一个问题,就是php.ini如果放在system32下,启 ...
- open()系统调用的实现
open系统调用的服务例程是sys_open()函数,它接受三个参数:要打开文件的路径名filename, 访问模式的表示flags和文件权限掩码mode.在内核中,sys_open实际调用do_sy ...
- Emacs的sr-speedbar中使能Go-mode
sr-speedbar使用了speedbar的文件检索功能,但是Emacs24自带的speedbar不支持go文件预览,下面是在speedbar中使能go-mode的一种方法: 1,按F10启动菜单栏 ...
- 【转】IOCP创建
转自:http://www.cmnsoft.com/wordpress/?p=248 感谢原作者.我在此整理一下: 完成端口(IOCP)是WINDOWS平台上特有的一种技术.要使用IOCP技术,就要用 ...
- mytop安装,使用mytop监控MySQL性能
本文主要描述mytop安装,安装过程中可能出现的报错,以及使用mytop监控MySQL性能. 欢迎转载,请注明作者.出处. 作者:张正 blog:http://space.itpub.net/2635 ...
- HDU 3018 Ant Trip (欧拉回路)
Ant Trip Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Su ...
- python中,花括号,中括号,小括号的区别
python中,花括号,中括号,小括号的区别 Python主要有三种数据类型:字典.列表.元组.其分别由花括号,中括号,小括号表示. 如: 字典:dic={'a':12,'b':34} 列表:list ...
- 一个MVC4 下的验证码用法
先看一个核心验证码类(不用在意实现过程,直接copy就行),下面包含了两种验证码图片(原理一样),代码如下: using System; using System.Collections.Generi ...
- 【Servlet】web.xml中welcome-file-list的作用
今天尝试使用struts2+ urlrewrite+sitemesh部署项目,结果发现welcome-file-list中定义的欢迎页不起作用: <welcome-file-list> & ...
- RVM Ruby 管理工具
1.RVM 简介 1.1 Ruby 简介 Ruby 是一种面向对象的脚本语言,简单易用,功能强大.能跨平台和可移植性好等等.其实就是种脚本语言. Ruby 的软件源使用的是亚马逊的云服务,国内网络环境 ...