android中的AsyncTask異步操作


      在開發Android移動客戶端的時候往往要使用多線程來進行操作,我們通常會將耗時的操作放在單獨的線程執行,避免其占用主線程而給用戶帶來不好的用戶 體驗。但是在子線程中無法去操作主線程(UI 線程),在子線程中操作UI線程會出現錯誤。因此android提供了一個類Handler來在子線程中來更新UI線程,用發消息的機制更新UI界面,呈 現給用戶。這樣就解決了子線程更新UI的問題。但是費時的任務操作總會啟動一些匿名的子線程,太多的子線程給系統帶來巨大的負擔,隨之帶來一些性能問題。 因此android提供了一個工具類AsyncTask,顧名思義異步執行任務。這個AsyncTask生來就是處理一些后台的比較耗時的任務,給用戶帶 來良好用戶體驗的,從編程的語法上顯得優雅了許多,不再需要子線程和Handler就可以完成異步操作並且刷新用戶界面。

先大概認識下Android.os.AsyncTask類:

       * android的類AsyncTask對線程間通訊進行了包裝,提供了簡易的編程方式來使后台線程和UI線程進行通訊:后台線程執行異步任務,並把操作結果通知UI線程。

       * AsyncTask是抽象類.AsyncTask定義了三種泛型類型 Params,Progress和Result。
   * Params 啟動任務執行的輸入參數,比如HTTP請求的URL。
   * Progress 后台任務執行的百分比。
    * Result 后台執行任務最終返回的結果,比如String,Integer等。

       * AsyncTask的執行分為四個步驟,每一步都對應一個回調方法,開發者需要實現這些方法。

   * 1) 繼承AsyncTask
    * 2) 實現AsyncTask中定義的下面一個或幾個方法
       * onPreExecute(), 該方法將在執行實際的后台操作前被UI 線程調用。可以在該方法中做一些准備工作,如在界面上顯示一個進度條,或者一些控件的實例化,這個方法可以不用實現。
       * doInBackground(Params...), 將在onPreExecute 方法執行后馬上執行,該方法運行在后台線程中。這里將主要負責執行那些很耗時的后台處理工作。可以調用 publishProgress方法來更新實時的任務進度。該方法是抽象方法,子類必須實現。
      * onProgressUpdate(Progress...),在publishProgress方法被調用后,UI 線程將調用這個方法從而在界面上展示任務的進展情況,例如通過一個進度條進行展示。
      * onPostExecute(Result), 在doInBackground 執行完成后,onPostExecute 方法將被UI 線程調用,后台的計算結果將通過該方法傳遞到UI 線程,並且在界面上展示給用戶.

      * onCancelled(),在用戶取消線程操作的時候調用。在主線程中調用onCancelled()的時候調用。

為了正確的使用AsyncTask類,以下是幾條必須遵守的准則:

      1) Task的實例必須在UI 線程中創建

      2) execute方法必須在UI 線程中調用

      3) 不要手動的調用onPreExecute(), onPostExecute(Result),doInBackground(Params...), onProgressUpdate(Progress...)這幾個方法,需要在UI線程中實例化這個task來調用。

      4) 該task只能被執行一次,否則多次調用時將會出現異常

      doInBackground方法和onPostExecute的參數必須對應,這兩個參數在AsyncTask聲明的泛型參數列表中指定,第一個為 doInBackground接受的參數,第二個為顯示進度的參數,第第三個為doInBackground返回和onPostExecute傳入的參 數。

 

下面通過一個Demo來說明如何使用Android.os.AsyncTask類,通過進度條來顯示進行的進度,程序結構圖如下:

[1] \layout\main.xml 布局文件源碼如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >
    <TextView
        android:id="@+id/tv1"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="@string/hello" />
    <ProgressBar
        android:id="@+id/pBar1"
        style="?android:attr/progressBarStyleHorizontal"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />
    <Button
        android:id="@+id/btn1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="開始異步操作" />
</LinearLayout>

 

[2] /src中的MainActivity.java源碼如下:

package lenghan.ma;

import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.ProgressBar;
import android.widget.TextView;

public class MainActivity extends Activity {
    /** Called when the activity is first created. */
    private TextView tv;
    private Button btn;
    private ProgressBar pbar;
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        tv=(TextView)findViewById(R.id.tv1);
        btn=(Button)findViewById(R.id.btn1);
        pbar=(ProgressBar)findViewById(R.id.pBar1);
        btn.setOnClickListener(
                new OnClickListener() {
                    public void onClick(View v) {
                        ProgressBarasyncTask asyncTask=new ProgressBarasyncTask(tv,pbar);
                        asyncTask.execute(1000);
                    }
                }
                );
    }
}

 

 [3] /src中的ProgressBarasyncTask.java源碼如下:

package lenghan.ma;

import android.os.AsyncTask;
import android.widget.ProgressBar;
import android.widget.TextView;
//生成該類的對象,並調用其execute方法之后
//首先執行的是onPreExecute方法
//其次是執行doInBackground方法
public class ProgressBarasyncTask extends AsyncTask<Integer, Integer, String>{
    private TextView tv;
    private ProgressBar pb;
    public ProgressBarasyncTask(TextView tv, ProgressBar pb) {
        this.tv = tv;
        this.pb = pb;
    }
    
    //該方法並不運行在UI線程內,所以在方法內不能對UI當中的控件進行設置和修改
    //主要用於進行異步操作
    @Override
    protected String doInBackground(Integer... params) {
            NetOperator netOperator =new NetOperator();
            int i=0;
            for ( i=10; i <=100; i+=10) {
                netOperator.operate();
                //執行publishProgress()調用onProgressUpdate()方法
                publishProgress(i);
            }
            return i+params[0].intValue()+"";
        }

    //該方法運行在Ui線程內,可以對UI線程內的控件設置和修改其屬性
    @Override
    protected void onPreExecute() {
        tv.setText("開始執行異步操作!");
    }

    //在doInBackground方法當中,每次調用publishProgrogress()方法之后,都會觸發該方法
    @Override
    protected void onProgressUpdate(Integer... values) {
        int value=values[0];
        pb.setProgress(value);
    }
    //在doInBackground方法執行結束后再運行,並且運行在UI線程當中
    //主要用於將異步操作任務執行的結果展示給用戶
    @Override
    protected void onPostExecute(String result) {
        tv.setText("異步操作執行結束"+result);
    }
    
//    @Override
//    protected void onCancelled() {
//        
//    }
 
    
}

 

 [4] /src中的NetOperator.java源碼如下:

package lenghan.ma;

public class NetOperator {

    public void operate() {
        // TODO Auto-generated method stub
        Thread th=new Thread();
        try {
            th.sleep(1000);
        } catch (InterruptedException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}

 [3] 下面看下程序的運行結果截圖:

參考:http://blog.csdn.net/cjjky/article/details/6684959


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM