asynctask的源碼
『壹』 asynctask 源碼,為什麼 android4.0 以後是串列
gridview和listview一樣,都是在你滑動的時候都會重復多次執行你的baseAdapter里的getView函數里代碼,你把AsyncTask放在getview里的話他當然會多次非同步載入了。
『貳』 android中實現非同步操作的方式有哪些,請簡述各自的特點
AsyncTask類。這是安卓自帶的非同步載入類,封裝較好,使用方便
直接開啟子線程通過handler發送消息更新,代碼繁瑣一點,AsyncTask源碼也是這個原理
『叄』 android.os.asynctask需要什麼jar包
在開發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傳入的參數。
下面通過一個Demo來說明如何使用Android.os.AsyncTask類,通過進度條來顯示進行的進度,然後用TextView來顯示進度值。程序結構圖如下:
[1] \layout\main.xml 布局文件源碼如下:
[html] view plain
<?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>
[html] view plain
<?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源碼如下:
[html] view plain
package com.andyidea.demo;
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 pb;
TextView tv;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
pb=(ProgressBar)findViewById(R.id.pb);
tv=(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++){
pb.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]來取值
tv.setText(progress[0]+"%");
super.onProgressUpdate(progress);
}
@Override
protected void onPostExecute(String result) {
//doInBackground返回時觸發,換句話說,就是doInBackground執行完後觸發
//這里的result就是上面doInBackground執行後的返回值,所以這里是"執行完畢"
setTitle(result);
super.onPostExecute(result);
}
}
}
[html] view plain
package com.andyidea.demo;
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 pb;
TextView tv;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
pb=(ProgressBar)findViewById(R.id.pb);
tv=(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++){
pb.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]來取值
tv.setText(progress[0]+"%");
super.onProgressUpdate(progress);
}
@Override
protected void onPostExecute(String result) {
//doInBackground返回時觸發,換句話說,就是doInBackground執行完後觸發
//這里的result就是上面doInBackground執行後的返回值,所以這里是"執行完畢"
setTitle(result);
super.onPostExecute(result);
}
}
}
『肆』 可編程式控制制器的點數僅指其輸入埠的點數,這句話對嗎
兩種情況,一種是運行結束,調用finish,一種是還沒結束,調用onProgressUpdate傳入進度值。是不是一下子就把三個方法串起來了。好了,到這里AsyncTask的源碼就分析完了。
總結下就是AsyncTask底層由Handler來完成線程的切換,內部由線程池來執行。有兩個線程池,一個用於任務排除,一個用於線程執行。
『伍』 Android AsyncTask源碼中 SerialExecutor類有什麼作用
Android中實現非同步任務機制有兩種方式,Handler與AsyncTask,之前我們分析過Handler消息處理機制(見:android-----Handler消息處理機制),今天來學一學AsyncTask,相對於採用handler模式為每個任務創建一個新的線程,任務完成之後通過Handler實例向UI線程發送消息,AsyncTask更加的重量級,他不需要編寫任務線程和Handler實例就可以完成相同的任務,但是他內部也是使用Handler來傳遞消息的,相當於是對Handler的封裝;
『陸』 在Android源碼中,AsyncTask中的onPostExecute方法是何時調用的
這個簡單, 一般要覆蓋三個方法, 1、onPreExecute(), 高負載代碼執行之前調用 ,通常用來顯示一個進度條,在主線程中執行 2、doInBackGround() : onPreExecute() 執行完後調用,此方法通常就是放高負載代碼的,比如遠程請求,巨大數據載入等,你不用新建線程來包裝此方法 AsyncTask(或子類)會自動在新線程中調用此方法 3、onPostExecute(Result), 在doInBackground完成之後調用,一般是設置結果,取消第一個方法顯示的進度條。 onProgressUpdate() 一般用來更新第一個方法顯示的進度條,什麼下載了50% 51% 。。。 總之,子類化AsyncTask 你不用顧及線程問題, 主線程中直接new AsyncTask的子類,並調用execute就行了,一定要在主線程中調execute。 還有,這些是AsyncTask的生命周期方法,你自己不要調用。