在開(kāi)發(fā)移動(dòng)客戶端的時(shí)候往往要使用多線程來(lái)進(jìn)行操作,我們通常會(huì)將耗時(shí)的操作放在單獨(dú)的線程執(zhí)行,避免其占用主線程而給用戶帶來(lái)不好的用戶體驗(yàn)。但是在子線程中無(wú)法去操作主線程(UI 線程),在子線程中操作UI線程會(huì)出現(xiàn)錯(cuò)誤。因此android提供了一個(gè)類Handler來(lái)在子線程中來(lái)更新UI線程,用發(fā)消息的機(jī)制更新UI界面,呈現(xiàn)給用戶。這樣就解決了子線程更新UI的問(wèn)題。但是費(fèi)時(shí)的任務(wù)操作總會(huì)啟動(dòng)一些匿名的子線程,太多的子線程給系統(tǒng)帶來(lái)巨大的負(fù)擔(dān),隨之帶來(lái)一些性能問(wèn)題。因此android提供了一個(gè)工具類AsyncTask,顧名思義異步執(zhí)行任務(wù)。這個(gè)AsyncTask生來(lái)就是處理一些后臺(tái)的比較耗時(shí)的任務(wù),給用戶帶來(lái)良好用戶體驗(yàn)的,從編程的語(yǔ)法上顯得優(yōu)雅了許多,不再需要子線程和Handler就可以完成異步操作并且刷新用戶界面。
AsyncTask是抽象類.AsyncTask定義了三種泛型類型 Params,Progress和Result。
Params 啟動(dòng)任務(wù)執(zhí)行的輸入?yún)?shù),比如HTTP請(qǐng)求的URL。
Progress 后臺(tái)任務(wù)執(zhí)行的百分比。
Result 后臺(tái)執(zhí)行任務(wù)最終返回的結(jié)果,比如String,Integer等。
AsyncTask的執(zhí)行分為四個(gè)步驟,每一步都對(duì)應(yīng)一個(gè)回調(diào)方法,開(kāi)發(fā)者需要實(shí)現(xiàn)這些方法。
1) 繼承AsyncTask
2) 實(shí)現(xiàn)AsyncTask中定義的下面一個(gè)或幾個(gè)方法
onPreExecute(), 該方法將在執(zhí)行實(shí)際的后臺(tái)操作前被UI 線程調(diào)用?梢栽谠摲椒ㄖ凶鲆恍(zhǔn)備工作,如在界面上顯示一個(gè)進(jìn)度條,或者一些控件的實(shí)例化,這個(gè)方法可以不用實(shí)現(xiàn)。
doInBackground(Params...), 將在onPreExecute 方法執(zhí)行后馬上執(zhí)行,該方法運(yùn)行在后臺(tái)線程中。這里將主要負(fù)責(zé)執(zhí)行那些很耗時(shí)的后臺(tái)處理工作?梢哉{(diào)用 publishProgress方法來(lái)更新實(shí)時(shí)的任務(wù)進(jìn)度。該方法是抽象方法,子類必須實(shí)現(xiàn)。
onProgressUpdate(Progress...),在publishProgress方法被調(diào)用后,UI 線程將調(diào)用這個(gè)方法從而在界面上展示任務(wù)的進(jìn)展情況,例如通過(guò)一個(gè)進(jìn)度條進(jìn)行展示。
onPostExecute(Result), 在doInBackground 執(zhí)行完成后,onPostExecute 方法將被UI 線程調(diào)用,后臺(tái)的計(jì)算結(jié)果將通過(guò)該方法傳遞到UI 線程,并且在界面上展示給用戶.
為了正確的使用AsyncTask類,以下是幾條必須遵守的準(zhǔn)則:
1) Task的實(shí)例必須在UI 線程中創(chuàng)建
2) execute方法必須在UI 線程中調(diào)用
3) 不要手動(dòng)的調(diào)用onPreExecute(), onPostExecute(Result),doInBackground(Params...), onProgressUpdate(Progress...)這幾個(gè)方法,需要在UI線程中實(shí)例化這個(gè)task來(lái)調(diào)用。
4) 該task只能被執(zhí)行一次,否則多次調(diào)用時(shí)將會(huì)出現(xiàn)異常
doInBackground方法和onPostExecute的參數(shù)必須對(duì)應(yīng),這兩個(gè)參數(shù)在AsyncTask聲明的泛型參數(shù)列表中指定,第一個(gè)為doInBackground接受的參數(shù),第二個(gè)為顯示進(jìn)度的參數(shù),第第三個(gè)為doInBackground返回和onPostExecute傳入的參數(shù)。
通過(guò)AsyncTask下載圖片示例:
/** * */ package com.demo.file; import java.io.IOException; import java.io.InputStream; import java.net.HttpURLConnection; import java.net.MalformedURLException; import java.net.URL; import android.app.Activity; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.os.AsyncTask; import android.os.Bundle; import android.view.View; import android.widget.ImageView; import android.widget.ProgressBar; import com.demo.HelloWorld.R; /** * @author xushilin xsl xushilin@kingtoneinfo.com * @version: 創(chuàng)建時(shí)間:2011-8-1 上午09:09:36 * 說(shuō) 明: 使用AsyncTask下載網(wǎng)絡(luò)數(shù)據(jù) * 修改歷史: */ public class DownLoadimgInAsyncTask extends Activity { private ImageView imgPic; private ProgressBar progressBar; private int totalSize = 0; private int size = 0; private String url = "/up/2011-8/201181233533157.jpg"; private Bitmap bitmap = null; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.download_image); imgPic = (ImageView) findViewById(R.id.imgPic); progressBar = (ProgressBar) findViewById(R.id.progressBar); progressBar.setProgress(getProgressInt(progressBar.getMax())); //實(shí)例化一個(gè)任務(wù)開(kāi)始執(zhí)行異步操作 myAncyncTask task = new myAncyncTask(); task.execute(url); } class myAncyncTask extends AsyncTask{ @Override protected Bitmap doInBackground(String... params) { // 這個(gè)方法是task中第二個(gè)被調(diào)用的方法,傳進(jìn)來(lái)的參數(shù)必須和AsyncTask 的 // 第一個(gè)泛型參數(shù)一直 URL fileURL = null; try { fileURL = new URL(params[0]); } catch (MalformedURLException err) { err.printStackTrace(); return null; } try { HttpURLConnection conn = (HttpURLConnection) fileURL .openConnection(); conn.setDoInput(true); conn.connect(); InputStream is = conn.getInputStream(); int length = (int) conn.getContentLength(); totalSize = length; if (length != -1) { byte[] imgData = new byte[length]; byte[] buffer = new byte[512]; int readLen = 0; int destPos = 0; while ((readLen = is.read(buffer)) > 0) { System.arraycopy(buffer, 0, imgData, destPos, readLen); destPos += readLen; size = destPos; publishProgress((int) ((size / (float) length) * 100)); Thread.sleep(100); } bitmap = BitmapFactory.decodeByteArray(imgData, 0, imgData.length); } return bitmap; } catch (IOException e) { e.printStackTrace(); return null; } catch (InterruptedException e) { e.printStackTrace(); return null; } } @Override protected void onPostExecute(Bitmap bitmap) { // 返回bitmap并展示在界面中,是最后一個(gè)被調(diào)用的方法 if (bitmap != null) imgPic.setImageBitmap(bitmap); progressBar.setVisibility(View.GONE); } @Override protected void onPreExecute() { // 這個(gè)方法是task中第一個(gè)被調(diào)用的方法 } @Override protected void onProgressUpdate(Integer... values) { // 更新進(jìn)度在doInBackground中調(diào)用publishProgress時(shí)這個(gè)會(huì)被調(diào)用,可以理解為第三個(gè)調(diào)用方法 System.out.println("" + values[0]); progressBar.setProgress(getProgressInt(values[0])); } } private int getProgressInt(int max) { int result = (totalSize > 0) ? (int) (size * max * 1.0 / totalSize) : 0; return result; } }
下載過(guò)程效果如下:
下載完成: