詳解Android 中AsyncTask 的使用
詳解Android 中AsyncTask 的使用
1、首先我們來看看AsyncTask 的介紹:
Handler 和 AsyncTask 都是android 中用來實現(xiàn)異步任務(wù)處理的方式;其中:
Handler 實例向 UI 線程發(fā)送消息,完成界面更新,
優(yōu)點:對整個過程控制的比較精細;
缺點:代碼相對臃腫,多個任務(wù)同時執(zhí)行時,不易對線程進行精確的控制;
AsyncTask :比Handler 更輕量級一些,適用于簡單的異步處理;
優(yōu)點:簡單 | 快捷 | 過程可控;
缺點:使用多個異步操作時就變得復(fù)雜起來;
2、AsyncTask 的定義:(AsyncTask 定義了三種泛型類型)
public abstract class AsyncTask<Params,Progress,Result>{...}
說明:
Params :啟動任務(wù)執(zhí)行的輸入?yún)?shù),例如:HTTP 請求的URL;
Progress: 后臺任務(wù)執(zhí)行的百分比;
Result:后臺執(zhí)行任務(wù)最終返回的結(jié)果,比如String;
3、AsyncTask 異步任務(wù)的執(zhí)行步驟:(以下方法除execute(Params... params),在AsyncTask中重寫),下列是相關(guān)方法的介紹:
A、execute(Params... params) :
執(zhí)行一個異步任務(wù),需要我們在UI線程中調(diào)用,觸發(fā)任務(wù)
B、OnPreExecute():
execute(Params... params)調(diào)用后立即執(zhí)行,一般用于在執(zhí)行后臺任務(wù)前對UI做一些標記; 例如,可以在此處顯示進度對話框;
C、doInBackground(Params.. params):
onPreExecute() 完成后執(zhí)行,后臺執(zhí)行,處理比較耗時的操作;此處不能操作UI,執(zhí)行的過程中調(diào)用publishProgress(Progress... values)來更新進度信息;
D、onProgressUpdate(Progress... values):
在調(diào)用publicshProgress(Progress... values)方法執(zhí)行,直接將進度信息更新到UI組建上;此方法在主線程上執(zhí)行,用于顯示任務(wù)執(zhí)行的進度;
E、onPostExecute(Result result):
此方法在主線程中執(zhí)行,當后臺的操作結(jié)束時,此方法會被調(diào)用,計算結(jié)果作為參數(shù)傳遞到此方法中,直接將結(jié)果顯示到UI組建上。
F、cancel(); :
取消一個正在執(zhí)行的任務(wù),在UI線程中完成,用AsyncTask的對象進行調(diào)用,參數(shù)為true/false;
4、使用AsyncTask 時注意事項:
A、異步任務(wù)實例必須在UI線程中創(chuàng)建;
B、execute(Params... params) 方法必須在UI線程中調(diào)用;
C、不要手動的調(diào)onPreExecute().doInBackground().onProgressUpdate().onPostExecute()這幾個方法;
D、不能在doInBackground(Params... params) 中更改組件信息;
E、一個任務(wù)實例只能執(zhí)行一次,如果執(zhí)行第二次會拋出異常;
5、案例:使用AsyncTask 實現(xiàn)圖片的下載:
Activity類,主程序的入口:
public class MainActivity extends Activity { // 程序入口 protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); MyAsyncTask my = new MyAsyncTask(); my.execute("http://photocdn.sohu.com/20110927/Img320705637.jpg"); } }
AsyncTask 派生類,實現(xiàn)異步任務(wù):
package com.sun.asynctask; import java.io.ByteArrayOutputStream; import java.io.InputStream; import java.net.HttpURLConnection; import java.net.URL; import org.apache.http.HttpConnection; import org.apache.http.HttpEntity; import org.apache.http.HttpResponse; import org.apache.http.HttpStatus; import org.apache.http.client.HttpClient; import org.apache.http.client.methods.HttpGet; import org.apache.http.impl.client.DefaultHttpClient; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.os.AsyncTask; import android.util.Log; /** * 異步任務(wù),實現(xiàn)網(wǎng)頁內(nèi)容的獲取 * * * 生成該類的對象,并調(diào)用execute方法之后 * * 首先執(zhí)行的是onProExecute() 方法, * * 其次執(zhí)行的doInBackground()方法 */ public class MyAsyncTask extends AsyncTask<String, Integer, Bitmap> { /** * 在execute() 方法執(zhí)行后立即執(zhí)行,運行在UI線程中, * 在后臺任務(wù)開始前執(zhí)行,用于標識UI界面 */ protected void onPreExecute() { super.onPreExecute(); Log.i("msg","onPreExecute()..."); } /** * 后臺執(zhí)行耗時的任務(wù); * * 方法中的 String 參數(shù)對應(yīng) AsyncTask的第一個參數(shù); * 返回的 Bitmap 對應(yīng)的是AsyncTask 的第三個參數(shù); * * 該方法并不運行在UI線程中,主要用于異步操作,可以調(diào)用publishProgress()方法觸發(fā) * onProgressUpdate對UI進行操作; * */ protected Bitmap doInBackground(String... params) { Log.i("msg","doInBackground(String... params)..."); try { /* 網(wǎng)絡(luò)訪問方式 二 */ /* URL url = new URL(params[0]); HttpsURLConnection connection = (HttpsURLConnection) url.openConnection(); connection.connect(); // 開始連接 int zong = connection.getContentLength(); InputStream is2 = connection.getInputStream(); */ /* 開始網(wǎng)絡(luò)訪問數(shù)據(jù) */ HttpGet hg = new HttpGet(params[0]); // 此處注意參數(shù)的用法 HttpClient hc = new DefaultHttpClient(); HttpResponse hr = hc.execute(hg); // 發(fā)送請求,得到響應(yīng) // 判斷請求是否成功 if(hr.getStatusLine().getStatusCode() == HttpStatus.SC_OK){ Log.i("msg", "access success..."); HttpEntity he = hr.getEntity(); InputStream is = he.getContent(); // 獲取輸入流對象,好比搭橋 long total = he.getContentLength(); // 文件的總字節(jié)數(shù) ByteArrayOutputStream baos = new ByteArrayOutputStream(); // 輸出流,臨時容器,用于裝從is中流出的數(shù)據(jù) byte[] buffer = new byte[1024]; // 緩存容器,每次裝載1024 個字節(jié)數(shù)據(jù) int len = 0; // 每次讀的字節(jié)數(shù) int curLen = 0 ; // 已讀多少數(shù)據(jù) while((len=is.read(buffer))!=-1){ // 當len !=-1 時,也就是還有數(shù)據(jù)可讀 Log.i("msg","begin read data..."+len+",total:"+total); baos.write(buffer, 0, len); // 向臨時容器中裝數(shù)據(jù) curLen=curLen+len; // 更新已讀的數(shù)據(jù) /* 在UI顯示當前讀取的進度 , 調(diào)用次方法觸發(fā)onProgressUpdate() 方法執(zhí)行 */ publishProgress((int)(((float)curLen/total)*100)); } Bitmap bitmap = BitmapFactory.decodeByteArray(baos.toByteArray(), 0, (int)total); is.close(); return bitmap; } } catch (Exception e) { e.printStackTrace(); } return null; } /** * 括號中的參數(shù):String 對應(yīng)的是AsyncTask 的第三個參數(shù),也就是 * 接收了 從doInBackground() 返回的結(jié)果; * 此方法在 doInBackground() 方法執(zhí)行結(jié)束后執(zhí)行,運行在UI線程中, * 可以對UI進行更新 */ protected void onPostExecute(Bitmap result) { super.onPostExecute(result); Log.i("msg","onPostExecute(String result)..."+result.getHeight()); } /** * 方法括號中的Integer 對應(yīng)AsyncTask 中的第二個參數(shù); * 在doInBackground() 中每次調(diào)用publishProgress() 時被執(zhí)行; * 該方法是在UI線程中的,所以可以用于對UI進行更新 */ protected void onProgressUpdate(Integer... values) { super.onProgressUpdate(values); Log.i("msg","onProgressUpdate(Integer... values)..."+values[0]); } /** * 圖片的下載 */ public HttpURLConnection downPic(String urltemp){ try { URL url = new URL(urltemp); // 確定連接地址 // 打開一個連接 HttpURLConnection connection = (HttpURLConnection) url.openConnection(); connection.connect(); // 開始連接 return connection; } catch (Exception e) { e.printStackTrace(); } return null; } }
以上就是Android AsyncTask的應(yīng)用實例,如有疑問請留言或者到本站社區(qū)交流討論,感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
- Android 使用AsyncTask實現(xiàn)多線程斷點續(xù)傳
- Android 使用AsyncTask實現(xiàn)斷點續(xù)傳
- Android 使用AsyncTask實現(xiàn)多任務(wù)多線程斷點續(xù)傳下載
- Android多線程AsyncTask詳解
- Android中使用AsyncTask實現(xiàn)下載文件動態(tài)更新進度條功能
- Android AsyncTask詳解及使用方法
- Android AsyncTask實現(xiàn)異步處理任務(wù)的方法詳解
- Android帶進度條的下載圖片示例(AsyncTask異步任務(wù))
- 淺談Android中AsyncTask的工作原理
相關(guān)文章
Android編程向服務(wù)器發(fā)送請求時出現(xiàn)中文亂碼問題的解決方法
這篇文章主要介紹了Android編程向服務(wù)器發(fā)送請求時出現(xiàn)中文亂碼問題的解決方法,實例分析了Android參數(shù)傳遞過程中中文亂碼的解決技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-11-11Android使用第三方服務(wù)器Bmob實現(xiàn)發(fā)送短信驗證碼
這篇文章主要介紹了Android使用第三方服務(wù)器Bmob實現(xiàn)發(fā)送短信驗證碼的思路詳解,需要的朋友可以參考下2016-09-09Android開發(fā)之微信底部菜單欄實現(xiàn)的幾種方法匯總
這篇文章主要介紹了Android開發(fā)之微信底部菜單欄實現(xiàn)的幾種方法,下面小編把每種方法通過實例逐一給大家介紹,需要的朋友可以參考下2016-09-09OpenGL關(guān)于glStencilFuncSeparate()和glStencilFunc()函數(shù)的區(qū)別講解
今天小編就為大家分享一篇OpenGL關(guān)于glStencilFuncSeparate()和glStencilFunc()函數(shù)的區(qū)別講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-04-04Android自定義ViewGroup實現(xiàn)豎向引導界面
這篇文章主要為大家詳細介紹了Andoird自定義ViewGroup實現(xiàn)豎向引導界面,具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-03-03Android 中讀取SD卡文件時拋出NullPointerException錯誤解決辦法
這篇文章主要介紹了Android 中讀取SD卡文件時拋出NullPointerException錯誤解決辦法的相關(guān)資料,需要的朋友可以參考下2017-05-05Android動畫之逐幀動畫(Frame Animation)基礎(chǔ)學習
大家都知道逐幀動畫是一種常見的動畫形式,其原理是在“連續(xù)的關(guān)鍵幀”中分解動畫動作,也就是在時間軸的每幀上逐幀繪制不同的內(nèi)容,使其連續(xù)播放而成動畫。下面我們就來學習下Android中逐幀動畫的基礎(chǔ)知識,有需要的可以參考借鑒。2016-09-09