AsyncTask 的使用方法網上有很多例子,使用起來也非常的方便。這里就不詳細說具體的使用方法了,同學可以Google 一下,很多。
場景模擬 當我們在加載一個列表的時候,比如GridView ,這時候我們考慮到不阻塞UI的做法,一般會使用線程Thread 、Timer 或者使用AsyncTask ,而這些操作都是在在后臺另外開一個線程給我們找數(shù)據,具體得到的數(shù)據需要使用Handler 去更新UI,AsyncTask 也是一樣使用到的Handler 只是它將Handler 封裝在了onPostExecute 執(zhí)行操作中。而這一操作可能會產生一個問題,比如你有一個列表更新數(shù)據庫使用到的是AsyncTask 異步操作的方式更新UI,而你的需求是當我一進來這個列表就統(tǒng)計這個列表的數(shù)據的數(shù)量或者讓某一行數(shù)據的狀態(tài)為選中狀態(tài)。傳統(tǒng)做法是直接new 一個AsyncTask 類讓它execute(); 之后會再操作UI。想法是對的,但有一個問題我們要注意到,因為它是異步加載數(shù)據的方式,而你的數(shù)據量比較多或許查找數(shù)據需要一定的時間的時候,這時使用AsyncTask 執(zhí)行異步加載后更新UI再操作UI對象,可能會報空指針。
這個問題的產生是,我們都知道代碼的執(zhí)行是自上而下執(zhí)行,當你使用異步加載數(shù)據的時候,代碼讓你去執(zhí)行異步操作就不管了(多線程),而繼續(xù)會往下執(zhí)行代碼,你下面的代碼就是操作列表里面的UI,這時可想而知,異步加載數(shù)據還沒有結束還沒有對你的UI進行更新,這些你的列表應該是空的,而操作一個空的列表就會報空指針。
分析問題
使用過AsyncTask 的同學都知道一個異步加載數(shù)據最少要重寫以下這兩個方法:
doInBackground 后臺執(zhí)行,比較耗時的操作都可以放在這里。注意這里不能直接操作UI。
onPostExecute 相當于Handler 處理UI的方式,在這里面可以使用在doInBackground 得到的結果處理操作UI。
有必要的話你還得重寫以下這三個方法,但不是必須的:
onProgressUpdate 可以使用進度條增加用戶體驗度。
onPreExecute 這里是最終用戶調用Excute時的接口
onCancelled 用戶調用取消時,要做的操作
根據上面的思路,可以看出,最終數(shù)據加載并顯示出來這一系列的操作都在onPostExecute 這個方法里面,那么如何監(jiān)聽所有UI都己經在onPostExecute 里面處理完成了,再去執(zhí)行我們自己要操作呢?
解決問題
這里給出我自己解決這一問題的思路,有更好想法的朋友歡迎跟貼共同探討。
首先創(chuàng)建一個接口
private interface isLoadDataListener {
public void loadComplete();
}
聲明這一接口變量
private isLoadDataListener loadLisneter;
給接口賦值,得到接口對象
public void setLoadDataComplete(isLoadDataListener dataComplete) {
this.loadLisneter = dataComplete;
}
之后在AsyncTask 的onPostExecute處理UI完成后調用該接口,下面給出一個我以前項目使用到的AsyncTask 類:
class loadGridAsyncTask extends AsyncTask《Integer, Integer, AppsAdapter》 {
private int poindex;
public loadGridAsyncTask(int positionindex) {
this.poindex = positionindex;
}
@Override
protected AppsAdapter doInBackground(Integer.。. params) {
// TODO Auto-generated method stub
// mAppsModel.clear();
Cursor temp = dbHelper.queryPageById(poindex);
loadPage(mApps, temp);
temp.close();
return new AppsAdapter(STB.this, mAppsModel);
}
@Override
protected void onPostExecute(AppsAdapter result) {
gridViewExt itemGrid = (gridViewExt) viewFlipper
.getChildAt(poindex);
itemGrid.setColumnCount(pageColumnCount);
itemGrid.setAdapter(result);
if (loadLisneter != null) {
loadLisneter.loadComplete();
}
}
}
AsyncTask 的使用方法網上有很多例子,使用起來也非常的方便。這里就不詳細說具體的使用方法了,同學可以Google 一下,很多。
場景模擬 當我們在加載一個列表的時候,比如GridView ,這時候我們考慮到不阻塞UI的做法,一般會使用線程Thread 、Timer 或者使用AsyncTask ,而這些操作都是在在后臺另外開一個線程給我們找數(shù)據,具體得到的數(shù)據需要使用Handler 去更新UI,AsyncTask 也是一樣使用到的Handler 只是它將Handler 封裝在了onPostExecute 執(zhí)行操作中。而這一操作可能會產生一個問題,比如你有一個列表更新數(shù)據庫使用到的是AsyncTask 異步操作的方式更新UI,而你的需求是當我一進來這個列表就統(tǒng)計這個列表的數(shù)據的數(shù)量或者讓某一行數(shù)據的狀態(tài)為選中狀態(tài)。傳統(tǒng)做法是直接new 一個AsyncTask 類讓它execute(); 之后會再操作UI。想法是對的,但有一個問題我們要注意到,因為它是異步加載數(shù)據的方式,而你的數(shù)據量比較多或許查找數(shù)據需要一定的時間的時候,這時使用AsyncTask 執(zhí)行異步加載后更新UI再操作UI對象,可能會報空指針。
這個問題的產生是,我們都知道代碼的執(zhí)行是自上而下執(zhí)行,當你使用異步加載數(shù)據的時候,代碼讓你去執(zhí)行異步操作就不管了(多線程),而繼續(xù)會往下執(zhí)行代碼,你下面的代碼就是操作列表里面的UI,這時可想而知,異步加載數(shù)據還沒有結束還沒有對你的UI進行更新,這些你的列表應該是空的,而操作一個空的列表就會報空指針。
分析問題
使用過AsyncTask 的同學都知道一個異步加載數(shù)據最少要重寫以下這兩個方法:
doInBackground 后臺執(zhí)行,比較耗時的操作都可以放在這里。注意這里不能直接操作UI。
onPostExecute 相當于Handler 處理UI的方式,在這里面可以使用在doInBackground 得到的結果處理操作UI。
有必要的話你還得重寫以下這三個方法,但不是必須的:
onProgressUpdate 可以使用進度條增加用戶體驗度。
onPreExecute 這里是最終用戶調用Excute時的接口
onCancelled 用戶調用取消時,要做的操作
根據上面的思路,可以看出,最終數(shù)據加載并顯示出來這一系列的操作都在onPostExecute 這個方法里面,那么如何監(jiān)聽所有UI都己經在onPostExecute 里面處理完成了,再去執(zhí)行我們自己要操作呢?
解決問題
這里給出我自己解決這一問題的思路,有更好想法的朋友歡迎跟貼共同探討。
首先創(chuàng)建一個接口
private interface isLoadDataListener {
public void loadComplete();
}
聲明這一接口變量
private isLoadDataListener loadLisneter;
給接口賦值,得到接口對象
public void setLoadDataComplete(isLoadDataListener dataComplete) {
this.loadLisneter = dataComplete;
}
之后在AsyncTask 的onPostExecute處理UI完成后調用該接口,下面給出一個我以前項目使用到的AsyncTask 類:
class loadGridAsyncTask extends AsyncTask《Integer, Integer, AppsAdapter》 {
private int poindex;
public loadGridAsyncTask(int positionindex) {
this.poindex = positionindex;
}
@Override
protected AppsAdapter doInBackground(Integer.。. params) {
// TODO Auto-generated method stub
// mAppsModel.clear();
Cursor temp = dbHelper.queryPageById(poindex);
loadPage(mApps, temp);
temp.close();
return new AppsAdapter(STB.this, mAppsModel);
}
@Override
protected void onPostExecute(AppsAdapter result) {
gridViewExt itemGrid = (gridViewExt) viewFlipper
.getChildAt(poindex);
itemGrid.setColumnCount(pageColumnCount);
itemGrid.setAdapter(result);
if (loadLisneter != null) {
loadLisneter.loadComplete();
}
}
}
通過上面的代碼,我們就得到一個數(shù)據加載完成后返回的接口,接下來的問題就是我們利用這個接口來處理我們的UI了,比如讓某一UI選中,得到這個列表的UI數(shù)量等,看下面的代碼:
new loadGridAsyncTask(1).execute();
setLoadDataComplete(new isLoadDataListener() {
@Override
public void loadComplete() {
// TODO Auto-generated method stub
//這里執(zhí)行你要的操作,當UI更新完成后會自動調用這里面的代碼 }
});
此篇文章希望能對入門不久的Android 開發(fā)者有幫助。