大家好日子。
所以,我有一个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仍然存在......
有什么想法吗?
感谢。
答案 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;
}