SQLiteDatabase使用过程中经常碰到的问题
1.DatabaseHelper在onCreate或者是onUpdate的时候调用DatabaseHelper.getWriteDatabase获取SQLiteDatabase
会导致一个递归死循环
解决方案:
如果在onCreate或者是onUpdate的时候需要获取一个SQLiteDatabase进行数据的读取与写入,应该在onCreate或者是onUpdate的时候设定这个过程中本身对应的SQLiteDatabase
2.多个线程中同时进行多张数据表的写(读数据是没有关系的)
写数据的使用,应该使用线程锁synchronized进行锁定,同时线程锁的对象应该是整个app中相同的database中完全一样的
3.多个线程中进行多张数据表的写,写完之后调用SQLiteDatabase.close进行了资源的关闭
例如,存在A,B,C,D...N多个线程,其中,B线程中在数据写完后进行了Databse.close操作
在A线程拿到线程锁进行写数据的时候,SQLiteDatabase会提示数据库已经关闭,不能进行数据的写入操作
解决方案:在获取writableDatabase的与readableDatabase的时候,应该通过函数来完成,例如:
private AtomicInteger mOpenCounter = new AtomicInteger(); /** * openDatabase与relaseDatabase需要配对出现 * @return */ public synchronized SQLiteDatabase openWriteDatabase() { final String userId = this.mCurrentUserId; DatabaseSQLiteHelper helper = getSQLiteHelper(userId); SQLiteDatabase db = getSQLiteHelper(userId).getDefaultWritableDatabase(); //db为空的情况或者是db已经关闭的情况 if(db == null || !db.isOpen()){ db = helper.getWritableDatabase(); helper.setDefaultWritableDatabase(db); } //每次调用一次,就将计数器往上加1次 incrementDatabseUsed(helper); return db; } /** * openDatabase与relaseDatabase需要配对出现 * @return */ public synchronized void relaseWriteDatabase(){ decrementWriteDatabseUsed(true); } private Object mAtomicLoker = new Object(); void incrementDatabseUsed(DatabaseSQLiteHelper helper){ synchronized (mAtomicLoker) { int result = mOpenCounter.incrementAndGet(); } } void decrementWriteDatabseUsed(boolean isCloseDatabse){ synchronized (mAtomicLoker) { final String userId = this.mCurrentUserId; if(TextUtils.isEmpty(userId)){ return; } DatabaseSQLiteHelper helper = getSQLiteHelper(userId); int result = mOpenCounter.decrementAndGet(); if(isCloseDatabse && !helper.isDoingDataMigratoring){ if(result <= 0){ SQLiteDatabase db = helper.getDefaultWritableDatabase(); if(db != null){ db.close(); } db = helper.getReadableDatabase(); if(db != null){ int currentResult = mOpenCounter.get(); if(currentResult <= 0){ db.close(); } } } } } }
相关推荐
Application的使用小总结 “call to OpenGL ES API with no current context (logged once per thread” 问题的解决 2013年9月7日19:15:33:我的平板分辨率很高可是运行public void onSurfaceChanged(GL10 gl, int ...
使用SQLiteDatabase数据库存储日记内容。 首先要有一个LoginActivity,输入密码,点击按钮,判断密码是否正确后,用intent跳转到MainActivity。 MainActivity主要包括activity_main里列表的关联,实现点击列表时跳转...
主要介绍了Android SQLite3多线程操作问题研究总结,本文总结了SQLite3是否支持多线程、SQLiteDatabase的同步锁、多线程读数据库等问题,需要的朋友可以参考下
7.3.3 SQLiteDatabase创建数据库的分析总结 7.4 Cursor 的query函数的实现分析 7.4.1 提取query关键点 7.4.2 MediaProvider 的query分析 7.4.3 query关键点分析 7.4.4 Cursor query实现分析总结 7.5 Cursor ...
SQLite SQLite是一种超轻量级的嵌入式数据库,大小只有几百KB,但是其语法支持标准SQL语法,同时还遵循了数据库的ACID事务,...getReadableDatabase() 获取SQLiteDatabase对象,操作数据库 getWritableDatabase()
目录: 实验一 熟悉Android的运行环境 实验二 UI设计 ...实验六 SQLite和SQLiteDatabase应用 实验七 多媒体应用 实验八 服务应用 (初学者的实验报告,仅供学习分享,谢绝商业转载,文件只有十个word文档)
/DATA/data/包名/databases是该程序存放数据的目录,DATA是...SQLiteDatabase db = openOrCreateDatabase(File file, SQLiteDatabase.Cursor, Factory factor); 2.SQLiteDatabase提供了如下方法: db.execSQL(s
这一节我将总结一下android中的另一种数据存储——SQLite 的相关知识点 SQLite数据库是android系统自带的,主要用到的类包括SQLiteOpenHelper和SQLiteDatabase。 1、SQLiteOpenHelper:创建数据库和数据库版本管理的...
市场上相应的检测平台诸如检测通、凡特网等皆为pc端检测网站,并且操作繁琐不够人性化,用户在实地使用中存在很多问题。昆山工业技术研究院着眼于为委托用户和质检机构搭建良好的沟通桥梁,免去目前市场业务中企业...