Android 異步任務(wù)AsyncTask學(xué)習(xí)

引言:搞 Android 這么久了,一直沒有主動(dòng)去學(xué)習(xí)使用 AsyncTask ,現(xiàn)在應(yīng)該很少有人在使用了,但面試中似乎總有人會(huì)問。最近一不小心在某項(xiàng)目中看到了相關(guān)的代碼,就決定對 AsyncTask 進(jìn)行一番學(xué)習(xí)。

時(shí)間:2016年11月20日11:51:41

作者:JustDo23

郵箱:JustDo_23@163.com

01. 背景

首先,Android 中子線程是不能進(jìn)行 UI 操作的。單線程操作避免了 UI 混亂的情況。其次,在主線程不能進(jìn)行耗時(shí)操作,否則會(huì)阻塞主線程,造成 ANR。因此,將耗時(shí)操作放在子線程成為了必然,子線程將處理的結(jié)果交給 UI 線程進(jìn)行 UI 更新,線程間通信必不可少。說到這里,最先想起來的就是 Android 中的 Handler 消息處理機(jī)制,除此之外 Android 中還封裝了一個(gè)抽象類 AsyncTaskAsyncTask其實(shí)是對HandlerThread的封裝,再者就是AsyncTask內(nèi)部是線程池實(shí)現(xiàn)。

在單線程模式中需要始終明確兩條:

  • UI 線程中能耗時(shí)操作,不能被阻塞
  • 非 UI 線程不能直接去更新 UI

02. 介紹

官方網(wǎng)站上介紹到AsyncTask是方便的線程操作,允許在后臺(tái)進(jìn)行操作并將結(jié)果返回給 UI 線程。建議在AsyncTask中執(zhí)行比較短的操作,如果是灰常灰常耗時(shí)的操作則強(qiáng)烈建議使用Executor等線程池進(jìn)行。一個(gè)異步任務(wù)定義3個(gè)泛型參數(shù)Params,Progress,Result,4個(gè)步驟onPreExecute,doInBackground,onProgressUpdate,onPostExecute;三個(gè)泛型就是啟動(dòng)參數(shù),任務(wù)進(jìn)度,返回結(jié)果,四個(gè)步驟就是啟動(dòng)準(zhǔn)備,后臺(tái)執(zhí)行,任務(wù)進(jìn)度,任務(wù)結(jié)果

03. 入門代碼

新建一個(gè)SimpleTask繼承AsyncTak,接著需要指定三個(gè)泛型,看到報(bào)錯(cuò)提示必須重寫doInBackground方法。然后繼續(xù)重寫其他的方法。

/**
 * 簡單使用
 *
 * @author JustDo23
 */
public class SimpleTask extends AsyncTask<Void, Void, Void> {

  /**
   * 異步操作執(zhí)行前初始化操作
   */
  @Override
  protected void onPreExecute() {
    super.onPreExecute();
    LogUtil.e("--->onPreExecute()");
  }

  /**
   * 異步執(zhí)行后臺(tái)線程將要完成的任務(wù)[這個(gè)是必須重寫的方法]
   *
   * @param params 泛型中的參數(shù)
   */
  @Override
  protected Void doInBackground(Void... params) {
    LogUtil.e("--->doInBackground()");
    publishProgress();// 進(jìn)行進(jìn)度更新
    return null;
  }

  /**
   * 異步任務(wù)[doInBackground]完成后系統(tǒng)自動(dòng)回調(diào)
   *
   * @param result [doInBackground]返回的結(jié)果
   */
  @Override
  protected void onPostExecute(Void result) {
    super.onPostExecute(result);
    LogUtil.e("--->onPostExecute()");
  }

  /**
   * 進(jìn)度更新 [doInBackground]方法中調(diào)用 publishProgress 方法后執(zhí)行
   *
   * @param progress
   */
  @Override
  protected void onProgressUpdate(Void... progress) {
    super.onProgressUpdate(progress);
    LogUtil.e("--->onProgressUpdate()");
  }

  @Override
  protected void onCancelled(Void aVoid) {
    super.onCancelled(aVoid);
  }

  @Override
  protected void onCancelled() {
    super.onCancelled();
  }

}

在 Activity 中的onCreate方法中進(jìn)行調(diào)用

new SimpleTask().execute();

打印出執(zhí)行步驟

E/JustDo23: --->onPreExecute()
E/JustDo23: --->doInBackground()
E/JustDo23: --->onProgressUpdate()
E/JustDo23: --->onPostExecute()

04. 異步加載網(wǎng)絡(luò)圖片

布局代碼

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:orientation="vertical"
  android:padding="20dp">

  <ImageView
    android:id="@+id/iv_net"
    android:layout_width="match_parent"
    android:layout_height="match_parent"/>

  <ProgressBar
    android:id="@+id/pb_loading"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_centerInParent="true"
    android:visibility="gone"/>

</RelativeLayout>

網(wǎng)絡(luò)加載代碼

/**
 * 展示圖片
 *
 * @author JustDo23
 */
public class ImageShowActivity extends Activity {

  private ImageView iv_net;// 圖片
  private ProgressBar pb_loading;// 加載圈

  private String imageUrl = "http://img4.duitang.com/uploads/item/201611/03/20161103224830_YGisc.thumb.700_0.jpeg";

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_image_show);
    iv_net = (ImageView) findViewById(R.id.iv_net);
    pb_loading = (ProgressBar) findViewById(R.id.pb_loading);

    new ImageTask().execute(imageUrl);// 在這里進(jìn)行調(diào)用
  }

  class ImageTask extends AsyncTask<String, Integer, Bitmap> {

    @Override
    protected void onPreExecute() {
      super.onPreExecute();
      pb_loading.setVisibility(View.VISIBLE);// 先將 loading 顯示
    }

    @Override
    protected Bitmap doInBackground(String... params) {
      String url = params[0];// 從參數(shù)中獲取網(wǎng)絡(luò)地址
      Bitmap bitmap = null;// 從網(wǎng)絡(luò)加載的圖片
      try {
        Thread.sleep(3000);// 以下是進(jìn)行網(wǎng)絡(luò)獲取圖片
        URLConnection urlConnection = new URL(url).openConnection();
        InputStream inputStream = urlConnection.getInputStream();
        BufferedInputStream bufferedInputStream = new BufferedInputStream(inputStream);
        bitmap = BitmapFactory.decodeStream(bufferedInputStream);
        inputStream.close();
        bufferedInputStream.close();
      } catch (Exception e) {
        e.printStackTrace();
      }
      return bitmap;// 將加載的圖片返回
    }

    @Override
    protected void onPostExecute(Bitmap bitmap) {
      super.onPostExecute(bitmap);
      pb_loading.setVisibility(View.GONE);// 將 loading 隱藏
      iv_net.setImageBitmap(bitmap);// 界面上設(shè)置顯示圖片
    }
  }
}

05. 進(jìn)度條顯示進(jìn)度

使用水平的進(jìn)度條和一個(gè) for 循環(huán)來模擬一下網(wǎng)絡(luò)加載進(jìn)度及界面的更新

/**
 * 利用 AsyncTask 實(shí)現(xiàn)進(jìn)度加載顯示
 *
 * @author JustDo23
 */
public class LoadingActivity extends Activity {

  private ProgressBar pb_loading;// 加載條

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.acitvity_loading);
    pb_loading = (ProgressBar) findViewById(R.id.pb_loading);

    new LoadingTask().execute();
  }

  class LoadingTask extends AsyncTask<Void, Integer, Void> {

    @Override
    protected Void doInBackground(Void... params) {
      try {// 循環(huán)模擬加載進(jìn)度
        for (int i = 0; i <= 100; i++) {
          LogUtil.e("progress = " + i);
          publishProgress(i);// 去更新界面
          Thread.sleep(300);
        }
      } catch (InterruptedException e) {
        e.printStackTrace();
      }
      return null;
    }

    @Override
    protected void onProgressUpdate(Integer... progress) {
      pb_loading.setProgress(progress[0]);
    }
  }
}

頻繁關(guān)閉啟動(dòng)界面,發(fā)現(xiàn)界面上的進(jìn)度條有時(shí)會(huì)一直沒有進(jìn)度;在日志打印過程中,發(fā)現(xiàn)AsyncTask從0到100的打印,一圈打印完了再打印第二圈。也就是AsyncTask內(nèi)部維護(hù)的 task 是串行的,一個(gè)執(zhí)行完了,再去執(zhí)行下一個(gè)。

06. 優(yōu)化

public class LoadingActivity extends Activity {

  private ProgressBar pb_loading;// 加載條
  private LoadingTask loadingTask;// 異步加載任務(wù)

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.acitvity_loading);
    pb_loading = (ProgressBar) findViewById(R.id.pb_loading);

    loadingTask = new LoadingTask();
    loadingTask.execute();
  }

  @Override
  protected void onPause() {
    super.onPause();
    if (loadingTask != null && loadingTask.getStatus() == AsyncTask.Status.RUNNING) {
      loadingTask.cancel(true);// cancel 方法只是將對應(yīng)的 AsyncTask 標(biāo)記為了取消狀態(tài),并不是真的取消線程執(zhí)行了。
    }
  }

  class LoadingTask extends AsyncTask<Void, Integer, Void> {

    @Override
    protected Void doInBackground(Void... params) {
      try {// 循環(huán)模擬加載進(jìn)度
        for (int i = 0; i <= 100; i++) {
          if (isCancelled()) {
            break;// 如果是取消狀態(tài)就直接跳出自動(dòng)結(jié)束線程
          }
          LogUtil.e("progress = " + i);
          publishProgress(i);// 去更新界面
          Thread.sleep(300);
        }
      } catch (InterruptedException e) {
        e.printStackTrace();
      }
      return null;
    }

    @Override
    protected void onProgressUpdate(Integer... progress) {
      if (isCancelled()) {
        return;// 如果是取消狀態(tài)就直接返回
      }
      pb_loading.setProgress(progress[0]);
    }
  }
}

注意:cancel 方法只是將對應(yīng)的 AsyncTask 標(biāo)記為了取消狀態(tài),并不是真的取消線程執(zhí)行了。

到這里,便對AsyncTask有了一個(gè)基礎(chǔ)的入門。以上的學(xué)習(xí)主要參考慕課網(wǎng)視頻Android必學(xué)-AsyncTask基礎(chǔ)

07. 串并行

網(wǎng)上有很多關(guān)于AsyncTask的串并行問題介紹,起初AsyncTask串行的,一個(gè)一個(gè)的執(zhí)行;接著修改成了并行,多線程并發(fā)執(zhí)行;接著又修改成了支持串行和并行,我們直接調(diào)用execute(Params... params)是進(jìn)行串行,調(diào)用executeOnExecutor(Executor exec, Params... params)是并傳遞類型,來選擇進(jìn)行并行還是串行。其實(shí)點(diǎn)開execute源碼

public final AsyncTask<Params, Progress, Result> execute(Params... params) {
        return executeOnExecutor(sDefaultExecutor, params);
}

從中可以看到是同一方法。參數(shù)有:AsyncTask.SERIAL_EXECUTORAsyncTask.THREAD_POOL_EXECUTOR;第一個(gè)是默認(rèn)的,串行操作;第二個(gè)是進(jìn)行并行操作。

08. 強(qiáng)調(diào)

  • 異步任務(wù)的實(shí)例必須在 UI 線程中創(chuàng)建
  • execute(Params... params)方法必須在 UI 線程中調(diào)用
  • 不要手動(dòng)回調(diào)onPreExecute(),doInBackground(Params... params),onPostExecute(Result result),onProgressUpdate(Progress... values)這四個(gè)方法
  • 不能在doInBackground(Params... params)方法中更新 UI
  • 一個(gè)任務(wù)實(shí)例只能執(zhí)行一次,第二次執(zhí)行就會(huì)拋出異常java.lang.IllegalStateException: Cannot execute task: the task is already running

09. 小綜合

參考 CSDN 上的文章詳解Android中AsyncTask的使用自己動(dòng)手寫了一個(gè)完整的 Demo

  • 在執(zhí)行AsyncTask.cancel(true);的時(shí)候先回調(diào)onCancelled()然后再回調(diào)onCancelled(Result result)點(diǎn)擊看下源碼onCancelled(Result result) 的內(nèi)部實(shí)現(xiàn)是直接調(diào)用onCancelled()
  • 在執(zhí)行AsyncTask.cancel(true);后,onPostExecute(Result result)方法是沒有進(jìn)行回調(diào)的
  • AsyncTask 的狀態(tài)被存放在一個(gè)枚舉Status中,總共有三種狀態(tài)FINISHED,PENDING,RUNNING
  • 點(diǎn)擊查看一下execute(Params... params)方法就會(huì)明白為啥只能執(zhí)行一次
public final AsyncTask<Params, Progress, Result> executeOnExecutor(Executor exec,  Params... params) {
    if (mStatus != Status.PENDING) {
        switch (mStatus) {
            case RUNNING:
                //如果該任務(wù)正在被執(zhí)行則拋出異常
                //值得一提的是,在調(diào)用cancel取消任務(wù)后,狀態(tài)仍未RUNNING
                throw new IllegalStateException("Cannot execute task:"
                        + " the task is already running.");
            case FINISHED:
                //如果該任務(wù)已經(jīng)執(zhí)行完成則拋出異常
                throw new IllegalStateException("Cannot execute task:"
                        + " the task has already been executed "
                        + "(a task can be executed only once)");
        }
    }

    //改變狀態(tài)為RUNNING
    mStatus = Status.RUNNING;

    //調(diào)用onPreExecute方法
    onPreExecute();

    mWorker.mParams = params;
    sExecutor.execute(mFuture);

    return this;
}

更多源碼解析可以到文章詳解Android中AsyncTask的使用中查看

文章推薦

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,321評論 6 543
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,559評論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,442評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,835評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,581評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,922評論 1 328
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,931評論 3 447
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 43,096評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,639評論 1 336
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,374評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,591評論 1 374
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,104評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,789評論 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,196評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,524評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,322評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,554評論 2 379

推薦閱讀更多精彩內(nèi)容