我在 SDCard 上有一个巨大的数据库 (40MB)。我需要LIKE
在查询中获取数据,这非常慢。
数据库请求大约需要 5 秒。因此,我需要异步并使用ProgressDialog
.
我试过了AsyncTask
,但问题出在ProgressDialog
.
它是这样实现的:
private class GetDataFromLangDB extends AsyncTask<String, String, String> {
private final ProgressDialog dialog = new ProgressDialog(TranslAndActivity.this);
@Override
protected void onPreExecute() {
super.onPreExecute();
urDBCursor.close();
curDBCursor = null;
scaAdapter = null;
this.dialog.setMessage("Loading data...");
this.dialog.show();
}
@Override
protected String doInBackground(String... whatSearch) {
String result = "";
if (myDatabaseAdapter != null) {
curDBCursor = myDatabaseAdapter.fetchAll(whatSearch[0]);
}
return result;
}
@Override
protected void onPostExecute(String result) {
super.onPostExecute(result);
if (this.dialog.isShowing()) {
this.dialog.dismiss();
}
prepareListView();
}
}
问题是ProgressDialog
在数据库请求期间未显示。完成数据库查询后,它会在屏幕上短暂闪烁。当用户在数据库请求期间尝试点击屏幕时,UI 被冻结,并且在数据库请求消息后显示“未响应”。
我用这种方式尝试了一个线程:
public void startProgress(View view, final String aWhatSearch) {
final ProgressDialog dialog = new ProgressDialog(MyActivity.this);
if (curDBCursor != null){
curDBCursor.close();
curDBCursor = null;
}
dialog.setMessage("Loading data...");
dialog.show();
Runnable runnable = new Runnable() {
public void run() {
curDBCursor = myDatabaseAdapter.fetchAll(aWhatSearch);
// dirty trick
try {
Thread.sleep(250); // it must be here to show progress
} catch (InterruptedException e) {
e.printStackTrace();
}
handler.post(new Runnable() {
public void run() {
if (dialog.isShowing()) {
dialog.dismiss();
}
prepareListView();
}
});
}
};
new Thread(runnable).start();
}
结果是一样的,但是当我使用 Thread.sleep(250)
;
的技巧时 ProgressDialog
在数据库请求期间显示。但它没有旋转,在 DB 请求期间它看起来冻结了。
数据库的东西是这样调用的(点击搜索按钮后):
btnSearchAll.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
// AsyncTask
new GetDataFromLangDB().execute(edtTextToSearch.getText().toString());
// or Thread
//startProgress(null, edtTextToSearch.getText().toString());
}
});
我在 SO 中发现了很多这样的问题,但没有什么对我有用。难道DB在SD卡上?