小米開原始檔管理器MiCodeFileExplorer-原始碼研究(5)-AsyncTask非同步任務

小雷FansUnion發表於2015-10-28

說明:本文的文字和程式碼,主要來自於網上的2篇文章。

  第4篇的時候,提到了非同步任務AsyncTask。

  網上找了2篇文章學習下,copy網友的程式碼,稍微改了幾個字,執行成功了。


  在開發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傳入的引數。


Activity原始碼

package cn.fansunion;
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 progreeBar;  
    TextView textView;  
      
    public void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.main);  
        progreeBar=(ProgressBar)findViewById(R.id.pb);  
        textView=(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>{  
        //後面尖括號內分別是引數(例子裡是執行緒休息時間),進度(publishProgress用到),返回值 型別  
          
        @Override  
        protected void onPreExecute() {  
            //第一個執行方法  
            super.onPreExecute();  
        }  
          
        @Override  
        protected String doInBackground(Integer... params) {  
            //第二個執行方法,onPreExecute()執行完後執行  
            for(int i=0;i<=100;i++){  
                progreeBar.setProgress(i);  
                publishProgress(i);  
                try {  
                    Thread.sleep(params[0]);  
                } catch (InterruptedException e) {  
                    e.printStackTrace();  
                }  
            }  
            return "執行完畢";  
        }  
  
        @Override  
        protected void onProgressUpdate(Integer... progress) {  
            //這個函式在doInBackground呼叫publishProgress時觸發,雖然呼叫時只有一個引數  
            //但是這裡取到的是一個陣列,所以要用progesss[0]來取值  
            //第n個引數就用progress[n]來取值  
            textView.setText(progress[0]+"%");  
            super.onProgressUpdate(progress);  
        }  
  
        @Override  
        protected void onPostExecute(String result) {  
            //doInBackground返回時觸發,換句話說,就是doInBackground執行完後觸發  
            //這裡的result就是上面doInBackground執行後的返回值,所以這裡是"執行完畢"  
            setTitle(result);  
            super.onPostExecute(result);  
        }  
          
    }  
}  

main.xml

<?xml version="1.0" encoding="utf-8"?>  
<LinearLayout xmlns:android="http://schemas.android.com/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="FansUnion,Android AsyncTask Demo "/>  
    <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="當前進度顯示"/>  
    <ProgressBar  
       android:id="@+id/pb"  
       android:layout_width="fill_parent"  
       android:layout_height="wrap_content"  
       style="?android:attr/progressBarStyleHorizontal"/>  
</LinearLayout>  



就一個類,看看文字,寫個Demo就可以了,特別需要的時候,再完整地看用法。


參考資料:

http://blog.csdn.net/xufenghappy6/article/details/7343899

http://www.cnblogs.com/devinzhang/archive/2012/02/13/2350070.html

相關文章