Android AsyncTask
在程序處理中必然會遇上耗時的操作,如訪問網絡,下載數據,訪問數據庫等,如何存在耗時的操作
又不能影響界面顯示交互。
在某些耗時可以控制的情況下,我們可以分批操作,對於事件循環執行;但有些情況如訪問網絡是沒法
控制的,處理過程屬於被動的。這類情況通常都是放在一個新的線程裏面執行。
Android中應用程序啓動後運行在主線程中(UI線程),處理界面交互,控件事件派發,界面設置顯示;
這些元素只能在UI線程操作。故不能在UI線程中處理耗時的動作,影響界面流暢操作,而出現ANR。
耗時的數據處理操作都需要在新的線程中後臺進行,基於線程的方式處理數據有很多形式:有Service,
Thread+Handler,AsyncTask。這裏看一下AsyncTask的使用。
AsyncTask:
更恰當和容易的使用UI線程,後臺操作都交給AsyncTask類完成,操作結果會返回給UI線程,這中間無需自己
控制Thread和Handler的操作。所以感覺似乎就是將Thread和Handler等複雜的事情封裝來,提供固定的模式進行
後臺操作,能更安全簡潔的進行後臺操作。
AsyncTask用法:
AsyncTask是一個抽象類。
public abstract class AsyncTask<Params, Progress, Result> {}
Here is an example of subclassing:
private class DownloadFilesTask extends AsyncTask<URL, Integer, Long> {
public void onPreExecute() {
super.onPreExecute();
}
protected Long doInBackground(URL... urls) {
int count = urls.length;
long totalSize = 0;
for (int i = 0; i < count; i++) {
totalSize += Downloader.downloadFile(urls[i]);
publishProgress((int) ((i / (float) count) * 100));
// Escape early if cancel() is called
if (isCancelled())
break;
}
return totalSize;
}
protected void onProgressUpdate(Integer... progress) {
setProgressPercent(progress[0]);
}
protected void onPostExecute(Long result) {
showDialog("Downloaded " + result + " bytes");
}
}
Once created, a task is executed very simply:
new DownloadFilesTask().execute(url1, url2, url3);
AsyncTask的泛型類型:
•Params 啓動任務執行時輸入參數,比如HTTP請求的URL。
•Progress 後臺任務執行的進度。
•Result 後臺執行任務最終返回的結果,比如String。
當然各參數的具體作用都可以是任意定義的。
AsyncTask執行四步:
- onPreExecute():Task被execute後被執行,可以爲UI線程中做一些準備工作:界面顯示。
- doInBackground:onPreExecute調用執行之後,由後臺線程調用。此方法運行在後臺線程中,
耗時的操作都是在這裏執行。使用publishProgress方法將執行的進度結果傳遞給UI線程,調用接口到onProgressUpdate。
- onProgressUpdate:在publishProgress方法被調用之後,由UI線程調用,傳遞後臺執行的進度,可以更新UI顯示錶明後臺操作進度。
- onPostExecute:後臺計算完成之後doInBackground執行完成之後,由UI線程中調用,傳遞後臺執行的結果。
取消Task:
Task可以被取消,通過cancel方法。取消之後onPostExecute不會被調用,而是onCancelled方法被調用。
AsyncTask 使用規則:
- AsyncTask必須在UI Thread中加載;
- AsyncTask實例必須在UI Thread中創建;
- Execute(Params…)方法必須在UI Thread中調用;
- 不要主動調用方法onPreExecute,doInBackground,onProgressUpdate,onPostExecute;
- Task只能被執行一次,之後再執行會異常。