AsyncTask使用的要點知識

在android開發中執行異步操作有handler和asyncTask兩種方法,使用handler要比asyncTask繁瑣一些,現在我們來接受如何使用asyncTask

在使用AsyncTask做異步任務的時候必須要遵循的原則:

  • AsyncTask類必須在UI Thread當中加載,在Android Jelly_Bean版本後這些都是自動完成的
  • AsyncTask的對象必須在UI Thread當中實例化
  • execute方法必須在UI Thread當中調用
  • 不要手動的去調用AsyncTask的onPreExecute, doInBackground, publishProgress, onProgressUpdate, onPostExecute方法,這些都是由Android系統自動調用的
  • AsyncTask任務只能被執行一次

一、Android當中的多線程

在Android當中,當一個應用程序的組件啓動的時候,並且沒有其他的應用程序組件在運行時,Android系統就會爲該應用程序組件開闢一個新的線程來執行。默認的情況下,在一個相同Android應用程序當中,其裏面的組件都是運行在同一個線程裏面的,這個線程我們稱之爲Main線程。當我們通過某個組件來啓動另一個組件的時候,這個時候默認都是在同一個線程當中完成的。當然,我們可以自己來管理我們的Android應用的線程,我們可以根據我們自己的需要來給應用程序創建額外的線程。

二、Main Thread 和 Worker Thread

在Android當中,通常將線程分爲兩種,一種叫做Main Thread,除了Main Thread之外的線程都可稱爲Worker Thread。

當一個應用程序運行的時候,Android操作系統就會給該應用程序啓動一個線程,這個線程就是我們的Main Thread,這個線程非常的重要,它主要用來加載我們的UI界面,完成系統和我們用戶之間的交互,並將交互後的結果又展示給我們用戶,所以Main Thread又被稱爲UI Thread。

Android系統默認不會給我們的應用程序組件創建一個額外的線程,所有的這些組件默認都是在同一個線程中運行。然而,某些時候當我們的應用程序需要完成一個耗時的操作的時候,例如訪問網絡或者是對數據庫進行查詢時,此時我們的UI Thread就會被阻塞。例如,當我們點擊一個Button,然後希望其從網絡中獲取一些數據,如果此操作在UI Thread當中完成的話,當我們點擊Button的時候,UI線程就會處於阻塞的狀態,此時,我們的系統不會調度任何其它的事件,更糟糕的是,當我們的整個現場如果阻塞時間超過5秒鐘(官方是這樣說的),這個時候就會出現 ANR (Application Not Responding)的現象,此時,應用程序會彈出一個框,讓用戶選擇是否退出該程序。對於Android開發來說,出現ANR的現象是絕對不能被允許的。

另外,由於我們的Android UI控件是線程不安全的,所以我們不能在UI Thread之外的線程當中對我們的UI控件進行操作。因此在Android的多線程編程當中,我們有兩條非常重要的原則必須要遵守:

  • 絕對不能在UI Thread當中進行耗時的操作,不能阻塞我們的UI Thread
  • 不能在UI Thread之外的線程當中操縱我們的UI元素

 三、如何處理UI Thread 和 Worker Thread之間的通信

既然在Android當中有兩條重要的原則要遵守,那麼我們可能就有疑問了?我們既不能在主線程當中處理耗時的操作,又不能在工作線程中來訪問我們的UI控件,那麼我們比如從網絡中要下載一張圖片,又怎麼能將其更新到UI控件上呢?這就關係到了我們的主線程和工作線程之間的通信問題了。在Android當中,提供了兩種方式來解決線程直接的通信問題,一種是通過Handler的機制(這種方式在後面的隨筆中將詳細介紹),還有一種就是今天要詳細講解的 AsyncTask 機制。

四、AsyncTask

AsyncTask:異步任務,從字面上來說,就是在我們的UI主線程運行的時候,異步的完成一些操作。AsyncTask允許我們的執行一個異步的任務在後臺。我們可以將耗時的操作放在異步任務當中來執行,並隨時將任務執行的結果返回給我們的UI線程來更新我們的UI控件。通過AsyncTask我們可以輕鬆的解決多線程之間的通信問題。

怎麼來理解AsyncTask呢?通俗一點來說,AsyncTask就相當於Android給我們提供了一個多線程編程的一個框架,其介於Thread和Handler之間,我們如果要定義一個AsyncTask,就需要定義一個類來繼承AsyncTask這個抽象類,並實現其唯一的一個 doInBackgroud 抽象方法。要掌握AsyncTask,我們就必須要一個概念,總結起來就是: 3個泛型,4個步驟。

3個泛型指的是什麼呢?我們來看看AsyncTask這個抽象類的定義,當我們定義一個類來繼承AsyncTask這個類的時候,我們需要爲其指定3個泛型參數:

AsyncTask <Params, Progress, Result>
  • Params: 這個泛型指定的是我們傳遞給異步任務執行時的參數的類型
  • Progress: 這個泛型指定的是我們的異步任務在執行的時候將執行的進度返回給UI線程的參數的類型
  • Result: 這個泛型指定的異步任務執行完後返回給UI線程的結果的類型

 我們在定義一個類繼承AsyncTask類的時候,必須要指定好這三個泛型的類型,如果都不指定的話,則都將其寫成Void,例如:

AsyncTask <Void, Void, Void>

4個步驟:當我們執行一個異步任務的時候,其需要按照下面的4個步驟分別執行

  • onPreExecute(): 這個方法是在執行異步任務之前的時候執行,並且是在UI Thread當中執行的,通常我們在這個方法裏做一些UI控件的初始化的操作,例如彈出要給ProgressDialog(doInBackground方法不會等這個函數運行完才運行,最好不要用這個方法做一些必要的初始化
  • doInBackground(Params... params): 在onPreExecute()方法執行完之後,會馬上執行這個方法,這個方法就是來處理異步任務的方法,Android操作系統會在後臺的線程池當中開啓一個worker thread來執行我們的這個方法,所以這個方法是在worker thread當中執行的,這個方法執行完之後就可以將我們的執行結果發送給我們的最後一個 onPostExecute 方法,在這個方法裏,我們可以從網絡當中獲取數據等一些耗時的操作
  • onProgressUpdate(Progess... values): 這個方法也是在UI Thread當中執行的,我們在異步任務執行的時候,有時候需要將執行的進度返回給我們的UI界面,例如下載一張網絡圖片,我們需要時刻顯示其下載的進度,就可以使用這個方法來更新我們的進度。這個方法在調用之前,我們需要在 doInBackground 方法中調用一個 publishProgress(Progress) 的方法來將我們的進度時時刻刻傳遞給 onProgressUpdate 方法來更新
  • onPostExecute(Result... result): 當我們的異步任務執行完之後,就會將結果返回給這個方法,這個方法也是在UI Thread當中調用的,我們可以將返回的結果顯示在UI控件上

 爲什麼我們的AsyncTask抽象類只有一個 doInBackground 的抽象方法呢??原因是,我們如果要做一個異步任務,我們必須要爲其開闢一個新的Thread,讓其完成一些操作,而在完成這個異步任務時,我可能並不需要彈出要給ProgressDialog,我並不需要隨時更新我的ProgressDialog的進度條,我也並不需要將結果更新給我們的UI界面,所以除了 doInBackground 方法之外的三個方法,都不是必須有的,因此我們必須要實現的方法是 doInBackground 方法。


asyncTask的一些缺點

使用AsyncTask的另一個問題是關於cancel。實際上,單單調用AsyncTask對象的cancel方法,並不能停止doInBackground方法的繼續執行。通常比較接受的方法是設置一個標誌位,也就是在每次執行前檢查一下某個變量的值(或者可以調用isCancelled方法判斷),來決定繼續執行還是停止。這種處理手段對於一些循環性的工作比較有用,但是對於一些循環性弱的工作可能並不怎麼有效。這也算是AsyncTask的一個弱點。和Thread相比,AsyncTask還有一個弱點是效率的問題,這個可以在本文開頭給出的鏈接中找到相關的信息。

 

AsyncTask還有一個問題和onPreExecute方法有關。這個方法是工作在UI線程的。雖然是叫onPreExecute,但是doInBackground方法(也就是實際上的execute),並不會等待onPreExecute方法做完全部操作纔開始執行。所以,一般還是不要用這個方法,可以在調用AsyncTask對象的execute方法之前就把該完成的操作完成,以免引起某些錯誤。

 

AsyncTask還有一個方法是onPostExecute,這個方法也是工作在UI線程,它是在doInBackground方法執行結束,並返回結果後調用。這個方法裏面可以調用UI線程的startActivity,這樣可以實現完成大量後臺操作後,自動跳轉Activity的功能。這個方法裏面也可以執行另一個AsyncTask的execute方法。



發佈了52 篇原創文章 · 獲贊 10 · 訪問量 14萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章