Android中AsyncTask的簡單用法

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

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

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

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

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

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

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

爲了正確的使用AsyncTask類,如下是幾條必須遵照的準則:app

  1) Task的實例必須在UI 線程中建立異步

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

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

  4) 該task只能被執行一次,不然屢次調用時將會出現異常

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

下面經過一個Demo來講明如何使用Android.os.AsyncTask類,經過進度條來顯示進行的進度,而後用TextView來顯示進度值。程序結構圖以下:

[1] \layout\main.xml 佈局文件源碼以下:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:orientation="vertical"
  4. android:layout_width="fill_parent"
  5. android:layout_height="fill_parent"
  6. >
  7. <TextView
  8. android:layout_width="fill_parent"
  9. android:layout_height="wrap_content"
  10. android:text="Hello , Welcome to Andy's Blog!"/>
  11. <Button
  12. android:id="@+id/download"
  13. android:layout_width="fill_parent"
  14. android:layout_height="wrap_content"
  15. android:text="Download"/>
  16. <TextView
  17. android:id="@+id/tv"
  18. android:layout_width="fill_parent"
  19. android:layout_height="wrap_content"
  20. android:text="當前進度顯示"/>
  21. <ProgressBar
  22. android:id="@+id/pb"
  23. android:layout_width="fill_parent"
  24. android:layout_height="wrap_content"
  25. style="?android:attr/progressBarStyleHorizontal"/>
  26. </LinearLayout>
<?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="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="當前進度顯示"/> <ProgressBar android:id="@+id/pb" android:layout_width="fill_parent" android:layout_height="wrap_content" style="?android:attr/progressBarStyleHorizontal"/> </LinearLayout>


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

  1. package com.andyidea.demo;
  2. import android.app.Activity;
  3. import android.os.AsyncTask;
  4. import android.os.Bundle;
  5. import android.view.View;
  6. import android.widget.Button;
  7. import android.widget.ProgressBar;
  8. import android.widget.TextView;
  9. public class MainActivity extends Activity {
  10. Button download;
  11. ProgressBar pb;
  12. TextView tv;
  13. /** Called when the activity is first created. */
  14. @Override
  15. public void onCreate(Bundle savedInstanceState) {
  16. super.onCreate(savedInstanceState);
  17. setContentView(R.layout.main);
  18. pb=(ProgressBar)findViewById(R.id.pb);
  19. tv=(TextView)findViewById(R.id.tv);
  20. download = (Button)findViewById(R.id.download);
  21. download.setOnClickListener(new View.OnClickListener() {
  22. @Override
  23. public void onClick(View v) {
  24. DownloadTask dTask = new DownloadTask();
  25. dTask.execute(100);
  26. }
  27. });
  28. }
  29. class DownloadTask extends AsyncTask<Integer, Integer, String>{
  30. //後面尖括號內分別是參數(例子裏是線程休息時間),進度(publishProgress用到),返回值 類型
  31. @Override
  32. protected void onPreExecute() {
  33. //第一個執行方法
  34. super.onPreExecute();
  35. }
  36. @Override
  37. protected String doInBackground(Integer... params) {
  38. //第二個執行方法,onPreExecute()執行完後執行
  39. for(int i=0;i<=100;i++){
  40. pb.setProgress(i);
  41. publishProgress(i);
  42. try {
  43. Thread.sleep(params[0]);
  44. } catch (InterruptedException e) {
  45. e.printStackTrace();
  46. }
  47. }
  48. return "執行完畢";
  49. }
  50. @Override
  51. protected void onProgressUpdate(Integer... progress) {
  52. //這個函數在doInBackground調用publishProgress時觸發,雖然調用時只有一個參數
  53. //可是這裏取到的是一個數組,因此要用progesss[0]來取值
  54. //第n個參數就用progress[n]來取值
  55. tv.setText(progress[0]+"%");
  56. super.onProgressUpdate(progress);
  57. }
  58. @Override
  59. protected void onPostExecute(String result) {
  60. //doInBackground返回時觸發,換句話說,就是doInBackground執行完後觸發
  61. //這裏的result就是上面doInBackground執行後的返回值,因此這裏是"執行完畢"
  62. setTitle(result);
  63. super.onPostExecute(result);
  64. }
  65. }
相關文章
相關標籤/搜索