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(); } } } } } }
?