在開(kāi)發(fā)Android移動(dòng)客戶端的時(shí)候往往要使用多線程來(lái)進(jìn)行操作,我們通常會(huì)將耗時(shí)的操作放在單獨(dú)的線程執(zhí)行,避免其占用主線程而給用戶帶來(lái)不好的用戶體驗(yàn)。但是在子線程中無(wú)法去操作主線程(UI 線程),在子線程中操作UI線程會(huì)出現(xiàn)錯(cuò)誤。因此android提供了一個(gè)類(lèi)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è)工具類(lèi)AsyncTask,顧名思義異步執(zhí)行任務(wù)。這個(gè)AsyncTask生來(lái)就是處理一些后臺(tái)的比較耗時(shí)的任務(wù),給用戶帶來(lái)良好用戶體驗(yàn)的,從編程的語(yǔ)法上顯得優(yōu)雅了許多,不再需要子線程和Handler就可以完成異步操作并且刷新用戶界面。
先大概認(rèn)識(shí)下Android.os.AsyncTask類(lèi):
* android的類(lèi)AsyncTask對(duì)線程間通訊進(jìn)行了包裝,提供了簡(jiǎn)易的編程方式來(lái)使后臺(tái)線程和UI線程進(jìn)行通訊:后臺(tái)線程執(zhí)行異步任務(wù),并把操作結(jié)果通知UI線程。
* AsyncTask是抽象類(lèi).AsyncTask定義了三種泛型類(lèi)型 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)度。該方法是抽象方法,子類(lèi)必須實(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 線程,并且在界面上展示給用戶.
* onCancelled(),在用戶取消線程操作的時(shí)候調(diào)用。在主線程中調(diào)用onCancelled()的時(shí)候調(diào)用。
為了正確的使用AsyncTask類(lèi),以下是幾條必須遵守的準(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ò)一個(gè)Demo來(lái)說(shuō)明如何使用Android.os.AsyncTask類(lèi),通過(guò)進(jìn)度條來(lái)顯示進(jìn)行的進(jìn)度,然后用TextView來(lái)顯示進(jìn)度值。程序結(jié)構(gòu)圖如下:

[1] \layout\main.xml 布局文件源碼如下:
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas./apk/res/android"
- android:orientation="vertical"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- >
- <TextView
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:text="Hello , Welcome to Andy's Blog!"/>
- <Button
- android:id="@+id/download"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:text="Download"/>
- <TextView
- android:id="@+id/tv"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:text="當(dāng)前進(jìn)度顯示"/>
- <ProgressBar
- android:id="@+id/pb"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- style="?android:attr/progressBarStyleHorizontal"/>
- </LinearLayout>
[2] /src中的MainActivity.java源碼如下:
- package com.andyidea.demo;
-
- import android.app.Activity;
- import android.os.AsyncTask;
- import android.os.Bundle;
- import android.view.View;
- import android.widget.Button;
- import android.widget.ProgressBar;
- import android.widget.TextView;
-
- public class MainActivity extends Activity {
-
- Button download;
- ProgressBar pb;
- TextView tv;
-
- /** Called when the activity is first created. */
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- pb=(ProgressBar)findViewById(R.id.pb);
- tv=(TextView)findViewById(R.id.tv);
-
- download = (Button)findViewById(R.id.download);
- download.setOnClickListener(new View.OnClickListener() {
- @Override
- public void onClick(View v) {
- DownloadTask dTask = new DownloadTask();
- dTask.execute(100);
- }
- });
- }
-
- class DownloadTask extends AsyncTask<Integer, Integer, String>{
- //后面尖括號(hào)內(nèi)分別是參數(shù)(例子里是線程休息時(shí)間),進(jìn)度(publishProgress用到),返回值 類(lèi)型
-
- @Override
- protected void onPreExecute() {
- //第一個(gè)執(zhí)行方法
- super.onPreExecute();
- }
-
- @Override
- protected String doInBackground(Integer... params) {
- //第二個(gè)執(zhí)行方法,onPreExecute()執(zhí)行完后執(zhí)行
- for(int i=0;i<=100;i++){
- pb.setProgress(i);
- publishProgress(i);
- try {
- Thread.sleep(params[0]);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- }
- return "執(zhí)行完畢";
- }
-
- @Override
- protected void onProgressUpdate(Integer... progress) {
- //這個(gè)函數(shù)在doInBackground調(diào)用publishProgress時(shí)觸發(fā),雖然調(diào)用時(shí)只有一個(gè)參數(shù)
- //但是這里取到的是一個(gè)數(shù)組,所以要用progesss[0]來(lái)取值
- //第n個(gè)參數(shù)就用progress[n]來(lái)取值
- tv.setText(progress[0]+"%");
- super.onProgressUpdate(progress);
- }
-
- @Override
- protected void onPostExecute(String result) {
- //doInBackground返回時(shí)觸發(fā),換句話說(shuō),就是doInBackground執(zhí)行完后觸發(fā)
- //這里的result就是上面doInBackground執(zhí)行后的返回值,所以這里是"執(zhí)行完畢"
- setTitle(result);
- super.onPostExecute(result);
- }
-
- }
- }
[3] 下面看下程序的運(yùn)行結(jié)果截圖: