小米開原始檔管理器MiCodeFileExplorer-原始碼研究(5)-AsyncTask非同步任務
說明:本文的文字和程式碼,主要來自於網上的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
相關文章
- 小米開原始檔管理器MiCodeFileExplorer-原始碼研究(0)-初步研究原始碼
- 小米開原始檔管理器MiCodeFileExplorer-原始碼研究(9)-入口分析原始碼
- 小米開原始檔管理器MiCodeFileExplorer-原始碼研究(1)-2個模型Model原始碼模型
- 小米開原始檔管理器MiCodeFileExplorer-原始碼研究(8)-檔案排序工具類FileSortHelper原始碼排序
- 小米開原始檔管理器MiCodeFileExplorer-原始碼研究(3)-使用最多的工具類Util原始碼
- 小米開原始檔管理器MiCodeFileExplorer-原始碼研究(2)-2個單例項工具類原始碼單例
- 小米開原始檔管理器MiCodeFileExplorer-原始碼研究(4)-檔案操作工具類FileOperationHelper原始碼
- 小米開原始檔管理器MiCodeFileExplorer-原始碼研究(6)-媒體檔案MediaFile和檔案型別MimeUtils原始碼型別
- 小米開原始檔管理器MiCodeFileExplorer-原始碼研究(7)-Favorite收藏管理和SQLite資料庫CRUD原始碼SQLite資料庫
- Android 非同步任務知識梳理(1) AsyncTask 原始碼解析Android非同步原始碼
- AsyncTask非同步任務類非同步
- Android中非同步任務(AsyncTask)Android非同步
- AsyncTask原始碼解析原始碼
- 深入理解安卓非同步任務AsyncTask安卓非同步
- AsyncTask實現非同步任務執行非同步
- Android的AsyncTask非同步任務淺析Android非同步
- Android 原始碼分析之 AsyncTask 原始碼分析Android原始碼
- Spring5原始碼解析-Spring中的非同步和計劃任務Spring原始碼非同步
- Free5GC原始碼研究(5) - NRF研究GC原始碼
- Android 非同步任務知識梳理(3) AsyncQueryHandler 原始碼解析Android非同步原始碼
- Android的AsyncTask原始碼分析Android原始碼
- Architecture(1)AsyncTask原始碼分析原始碼
- Android執行緒管理之AsyncTask非同步任務Android執行緒非同步
- 懸賞任務系統原始碼_任務接單平臺原始碼PHP版附app原始碼PHPAPP
- Android開發AsyncTask原始碼分析【模板方法模式】Android原始碼模式
- gulp原始碼解析之任務管理原始碼
- gulp原始碼解析(三)—— 任務管理原始碼
- Android 非同步任務知識梳理(2) HandlerThread 原始碼解析Android非同步thread原始碼
- OkHttpClient原始碼分析(一)—— 同步、非同步請求分析和Dispatcher的任務排程HTTPclient原始碼非同步
- Free5GC原始碼研究(4) - AUSF研究GC原始碼
- LiteOS-任務篇-原始碼分析-任務排程函式原始碼函式
- spark原始碼之任務提交過程Spark原始碼
- 懸賞任務小程式原始碼(開源php系統)原始碼PHP
- Free5GC原始碼研究(9) - PCF研究(下)GC原始碼
- 【原始碼解析】AsyncTask的用法與規則原始碼
- 原始碼分析 —— AsyncTask 完全解析(基於7.0)原始碼
- MapReduce——客戶端提交任務原始碼分析客戶端原始碼
- LiteOS核心原始碼分析:任務LOS_Schedule原始碼