AsyncTask上的android Sqlite IllegalStateException

时间:2014-06-13 16:02:39

标签: android sqlite android-asynctask illegalstateexception

大家好日子。

所以,我有一个AsyncTask从图像计算高斯模糊。使用Blob字段从SQLite检索原始图像数据,也将过滤后的图像存储在SQLite中。这是代码(setQuizData是执行此操作的函数):

public class SaveFilterTask  extends AsyncTask<Void, Void, Void>{

private Context mContext;
private Context mApplicationContext;
private Quiz mQuiz;     

public SaveFilterTask(Context context, Quiz quiz, Context applicationContext) {
    super();
    this.mContext = context;
    this.mApplicationContext = applicationContext;
    this.mQuiz = quiz;
}

@Override
protected Void doInBackground(Void... arg0) {
    try {
        setQuizData(mQuiz);
    } catch (UnsupportedEncodingException e) {
        cancel(true);
    }
    return null;
}

@Override
protected void onCancelled() {
    Toast toast = Toast.makeText(mContext, R.string.error_loading_quiz, Toast.LENGTH_SHORT);
    toast.setGravity(Gravity.CENTER, 0, 0);
    toast.show();
}

@Override
protected void onPreExecute() {
    if(!((Activity)mContext).isFinishing()){
        ((MainActivity)mContext).showDialog();
    }
}

protected void onPostExecute(Void args) {

    if(!((Activity)mContext).isFinishing()){
        ((MainActivity)mContext).hideDialog();
    }

    Intent solveQuizIntent = new Intent(mContext, SolveQuizActivity.class);
    solveQuizIntent.putExtra(
            QuizConstants.KEY_PARCELABLE_FINISHED_QUIZ, mQuiz);
    solveQuizIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    mContext.startActivity(solveQuizIntent);
}

private void setQuizData(Quiz quiz) throws UnsupportedEncodingException {
    if(quiz.getType().equals(QuizConstants.TYPE_PHOTO)){
        QuizDataSource quizdatasource = new QuizDataSource(mApplicationContext);
        quizdatasource.open();
        String data = quizdatasource.getData(quiz.getId());
        quizdatasource.close();
        QuizDataSource quizdatasource2 = new QuizDataSource(mApplicationContext);
        quizdatasource2.open();
        String filterData = quizdatasource2.getDataFilter(quiz.getId());
        quizdatasource2.close();

        String filter = quiz.getFilter();
        if(filter != null){
            if(!filter.equals(QuizConstants.FILTER_DEFAULT)){
                Bitmap original = decodeImage(data);                                    
                if(filterData == null){         
                    data = FilterManager.applyFilter(original, quiz.getFilter());
                    Log.d("FilterManager","Data: "+ data);
                    Log.d("FilterManager","Id: "+ quiz.getId());
                    QuizDataSource quizdatasourceW = new QuizDataSource(mApplicationContext);
                    quizdatasourceW.open();
                    quizdatasourceW.setDataFilter(quiz.getId(), data);
                    quizdatasourceW.close();
                }                   
            }
        }           
    }
}


private static Bitmap decodeImage(String data) {         
    byte[] b = Base64.decode(data, Base64.DEFAULT);                 
    return BitmapFactory.decodeByteArray(b, 0, b.length);
}

}

从我的Activity中调用此任务,如下所示:

SaveFilterTask sftask = new SaveFilterTask(this, quiz, getApplicationContext());
sftask.execute();

我不知道为什么有时(并随机)我会收到此错误:

0java.lang.RuntimeException: An error occured while executing doInBackground()
1at android.os.AsyncTask$3.done(AsyncTask.java:300)
2at java.util.concurrent.FutureTask.finishCompletion(FutureTask.java:355)
3at java.util.concurrent.FutureTask.setException(FutureTask.java:222)
4at java.util.concurrent.FutureTask.run(FutureTask.java:242)
5at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:231)
6at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
7at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:587)
8at java.lang.Thread.run(Thread.java:841)
9Caused by: java.lang.IllegalStateException: attempt to re-open an already-closed object: SQLiteDatabase: /data/data/Data.db
10at android.database.sqlite.SQLiteClosable.acquireReference(SQLiteClosable.java:55)
11at android.database.sqlite.SQLiteDatabase.queryWithFactory(SQLiteDatabase.java:1156)
12at android.database.sqlite.SQLiteDatabase.query(SQLiteDatabase.java:1032)
13at android.database.sqlite.SQLiteDatabase.query(SQLiteDatabase.java:1200)
14at QuizDataSource.getDataFilter(QuizDataSource.java:119)
15at SaveFilterTask.setQuizData(SaveFilterTask.java:82)
16at SaveFilterTask.doInBackground(SaveFilterTask.java:39)
17at SaveFilterTask.doInBackground(SaveFilterTask.java:1)

我尝试了不同的方式,使用单个连接,一个连接用于读取操作,另一个用于写入操作。但是这个bug仍然存在......

有什么想法吗?

感谢。

1 个答案:

答案 0 :(得分:0)

首先,你不应该每次都重新初始化QuazDataSource。

在Construcotr中初始化它并使用

进行检查
if (quizdatasource  == null) {
    quizdatasource = CreateaQuizDataSourceHelper().open;
}

然后你应该确保关闭它,即使它失败了。

Cursor cursor = null;
try {
  if (quizdatasource  == null) {
 quizdatasource = CreateaQuizDataSourceHelper().open;
  } 
  cursor= quizdatasource.rawQuery(...);
} catch (Exception e) { e.printStackTrace(); }
finally {
  if (quizdatasource != null) { quizdatasource.close(); }
}

至少确保你的游标关闭,并将验证放在finally块中。

if (cursor != null) { cursor.close; } 

然后你不应该得到多个实例。

您还应该创建数据库的SINGLETON实例,以确保只访问onec。

Helper本身可能有一个开放的方法来确保它使用相同的连接。

    public SQLiteDatabase open() {
    SQLiteDatabase sqLiteDatabase = null;
    try {

        try {
            sqLiteDatabase = this.getDatabaseConnection();
        } catch (SQLiteException e) {
            e.printStackTrace();
            DataBaseConnectionPool.DbHelper DbHelper = this.dbHelper;
            if (DbHelper != null) {
                return this.dbHelper.getReadableDatabase();
            }
        }

    } catch (Exception e) {
        e.printStackTrace();
    }
    return sqLiteDatabase;
}