Android使用AsyncTask 有如下好處:
1. 線程的開銷較大,如果每個任務都要創建一個線程,那麼應用程序的效率要低很多;
2. 線程無法管理,匿名線程創建並啓動後就不受程序的控制了,如果有很多個請求發送,那麼就會啓動非常多的線程,系統將不堪重負;
3. 另外,在新線程中更新UI還必須要引入handler,這讓代碼看上去非常臃腫;
爲了解決這一問題,Android在1.5版本引入了AsyncTask. AsyncTask的特點是任務在主線程之外運行,而回調方法是在主線程中執行,這就有效地避免了使用Handler帶來的麻煩。閱讀AsyncTask的源碼可知,AsyncTask是使用java.util.concurrent 框架來管理線程以及任務的執行的,concurrent框架是一個非常成熟,高效的框架,經過了嚴格的測試。這說明AsyncTask的設計很好的解決了匿名線程存在的問題。
AsyncTask定義了三種泛型類型 Params,Progress和Result.
- Params啓動任務執行的輸入參數,比如HTTP請求的URL;
- Progress後臺任務執行的百分比;
- Result後臺執行任務最終返回的結果,比如String;
子類必須實現抽象方法doInBackground(Params… p) ,在此方法中實現任務的執行工作,比如連接網絡獲取數據等。通常還應該實現onPostExecute(Result r)方法,因爲應用程序關心的結果在此方法中返回。需要注意的是AsyncTask一定要在主線程中創建實例。
AsyncTask的執行分爲四個步驟,每一步都對應一個回調方法,需要注意的是這些方法不應該由應用程序調用,開發者需要做的就是實現這些方法。在任務的執行過程中,這些方法被自動調用,運行過程,
onPreExecute()當任務執行之前開始調用此方法,可以在這裏顯示進度對話框。
doInBackground(Params…)此方法在後臺線程執行,完成任務的主要工作,通常需要較長的時間。在執行過程中可以調用publicProgress(Progress…)來更新任務的進度。
onProgressUpdate(Progress…)此方法在主線程執行,用於顯示任務執行的進度。
onPostExecute(Result)此方法在主線程執行,任務執行的結果作爲此方法的參數返回。
class GetImageTask extends AsyncTask<String, Void, Bitmap>{
InputStream is = null;
@Override
protected Bitmap doInBackground(String... params) {
// TODO Auto-generated method stub
URL myFileUrl = null;
Bitmap bitmap = null;
InputStream is = null;
HttpURLConnection conn = null;
try {
myFileUrl = new URL(params[0]);
} catch (MalformedURLException e) {
e.printStackTrace();
}
try {
conn = (HttpURLConnection)myFileUrl
.openConnection();
conn.setDoInput(true);
conn.connect();
is =conn.getInputStream();
bitmap =BitmapFactory.decodeStream(is);
is.close();
} catch (IOException e) {
e.printStackTrace();
}finally{
try {
if(is != null){
is.close();
}
if( conn != null){
conn.disconnect();
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
return bitmap;
}
@Override
protected void onCancelled() {
// TODO Auto-generated method stub
super.onCancelled();
}
@Override
protected void onPostExecute(Bitmap result) {
// TODO Auto-generated method stub
mImageAndTextView.setImage(result);
mImageAndTextView.postInvalidate(0, 0 , mScreenWidth ,mImageHeight + 30); //只更新稍比圖片大一些的區域
super.onPostExecute(result);
}
}