亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

Android AsyncTask與handler

系統(tǒng) 2140 0

本文主要講解下AsyncTask的使用以及Handler的應(yīng)用

首先,我們得明確下一個概念,什么是UI線程。顧名思義,ui線程就是管理著用戶界面的那個線程!

android的ui線程操作并不是安全的,并且和用戶直接進(jìn)行界面交互的操作都必須在ui線程中進(jìn)行才可以。這種模式叫做單線程模式。

我們在單線程模式下編程一定要注意: 不要阻塞ui線程、確保只在ui線程中訪問ui組件

當(dāng)我們要執(zhí)行一個復(fù)雜耗時的算法并且最終要將計算結(jié)果反映到ui上時,我們會發(fā)現(xiàn),我們根本沒辦法同時保證上面的兩點要求;我們肯定會想到開啟一個新的線程,讓這個復(fù)雜耗時的任務(wù)到后臺去執(zhí)行,但是執(zhí)行完畢了呢?我們發(fā)現(xiàn),我們無法再與ui進(jìn)行交互了。

為了解決這種情況,android為我們提供了很多辦法。

1)、handler和message機制:通過顯示的拋出、捕獲消息與ui進(jìn)行交互;

2)、Activity.runOnUiThread(Runnable):如果當(dāng)前線程為ui線程,則立即執(zhí)行;否則,將參數(shù)中的線程操作放入到ui線程的事件隊列中,等待執(zhí)行。

3)、View.post(Runnable):將操作放入到message隊列中,如果放入成功,該操作將會在ui線程中執(zhí)行,并返回true,否則返回false

4)、View.postDelayed(Runnable, long)跟第三條基本一樣,只不過添加了一個延遲時間。

5)、android1.5以后為我們提供了一個工具類來搞定這個問題AsyncTask.


AsyncTask是抽象類 ,定義了三種泛型類型 Params,Progress,Result。

Params 啟動任務(wù)執(zhí)行的輸入?yún)?shù),比如HTTP請求的URL

Progress 后臺任務(wù)執(zhí)行的百分比。

Result 后臺執(zhí)行任務(wù)最終返回的結(jié)果,比如String

用程序調(diào)用,開發(fā)者需要做的就是實現(xiàn)這些方法。

1) 子類化AsyncTask

2) 實現(xiàn)AsyncTask中定義的下面一個或幾個方法

onPreExecute() ,該方法將在執(zhí)行實際的后臺操作前被UI thread調(diào)用。可以在該方法中做一些準(zhǔn)備工作,如在界面上顯示一個進(jìn)度條。

doInBackground(Params…) ,將在onPreExecute 方法執(zhí)行后馬上執(zhí)行,該方法運行在后臺線程中。這里將主要負(fù)責(zé)執(zhí)行那些很耗時的后臺計算工作。可以調(diào)用 publishProgress方法來更新實時的任務(wù)進(jìn)度。該方法是抽象方法,子類必須實現(xiàn)。

onProgressUpdate(Progress…) ,在publishProgress方法被調(diào)用后,UI thread將調(diào)用這個方法從而在界面上展示任務(wù)的進(jìn)展情況,例如通過一個進(jìn)度條進(jìn)行展示。

onPostExecute(Result) ,在doInBackground 執(zhí)行完成后,onPostExecute 方法將被UI thread調(diào)用,后臺的計算結(jié)果將通過該方法傳遞到UI thread.

為了正確的使用AsyncTask類,以下是幾條必須遵守的準(zhǔn)則:

1) Task的實例必須在UI thread中創(chuàng)建

2) execute方法必須在UI thread中調(diào)用

3) 不要手動的調(diào)用onPreExecute(), onPostExecute(Result),doInBackground(Params…), onProgressUpdate(Progress…)這幾個方法

4) 該task只能被執(zhí)行一次,否則多次調(diào)用時將會出現(xiàn)異常

    package cn.com.chenzheng_java;
 
import android.os.AsyncTask;
/**
 * 
 * @author chenzheng_java
 * @description 異步任務(wù)AcyncTask示例
 *    
 */
public class MyAsyncTask extends AsyncTask<String, Integer, Object> {
 
 /**
 * 該方法由ui線程進(jìn)行調(diào)用,用戶可以在這里盡情的訪問ui組件。
 * 很多時候,我們會在這里顯示一個進(jìn)度條啥的,以示后臺正在
 * 執(zhí)行某項功能。
 */
 @Override
 protected void onPreExecute() {
 super.onPreExecute();
 }
 
 /**
 * 該方法由后臺進(jìn)程進(jìn)行調(diào)用,進(jìn)行主要的耗時的那些計算。
 * 該方法在onPreExecute方法之后進(jìn)行調(diào)用。當(dāng)然在執(zhí)行過程中
 * 我們可以每隔多少秒就調(diào)用一次publishProgress方法,更新
 * 進(jìn)度信息
 */
 @Override
 protected Object doInBackground(String... params) {
 return null;
 }
 
 
 /**
 * doInBackground中調(diào)用了publishProgress之后,ui線程就會
 * 調(diào)用該方法。你可以在這里動態(tài)的改變進(jìn)度條的進(jìn)度,讓用戶知道
 * 當(dāng)前的進(jìn)度。
 */
 @Override
 protected void onProgressUpdate(Integer... values) {
 super.onProgressUpdate(values);
 }
 
 /**
 * 當(dāng)doInBackground執(zhí)行完畢之后,由ui線程調(diào)用。可以在這里
 * 返回我們計算的最終結(jié)果給用戶。
 */
 @Override
 protected void onPostExecute(Object result) {
 super.onPostExecute(result);
 }
}
  


下面介紹最本質(zhì)的多線程:hanlder和message機制:

為何需要多線程:

在日常應(yīng)用中,我們通常需要處理一些“后臺,用戶不可見”的操作,例如說,我們需要下載一個音樂,要是你的應(yīng)用必須等用戶下載完成之后才可以進(jìn)行別的操作,那肯定讓用戶非常的不爽。這時候,我們通常的做法是,讓這些操作去后臺執(zhí)行,然后等后臺執(zhí)行完畢之后,再給用戶彈出相應(yīng)的提示信息。這時候,我們就需要使用多線程機制,然后通過創(chuàng)建一個新的線程來執(zhí)行這些操作。

明白了,實現(xiàn)需求,我們就準(zhǔn)備著手實現(xiàn)了。但是,經(jīng)過進(jìn)一步的了解,我們悲劇的發(fā)現(xiàn),android中的線程機制是,只能在UI線程中和用戶進(jìn)行交互。當(dāng)我們創(chuàng)建了一個新線程,執(zhí)行了一些后臺操作,執(zhí)行完成之后,我們想要給用戶彈出對話框以確認(rèn),但是卻悲劇的發(fā)現(xiàn),我們根本無法返回UI主線程了。 (UI線程就是你當(dāng)前看到的這些交互界面所屬的線程)。

這時候,我們?nèi)绻胍獙崿F(xiàn)這些功能,我們就需要一個android為我們提供的handler和message機制。

先講解下編程機制:

我們通常 在UI線程中創(chuàng)建一個handler,handler相當(dāng)于一個處理器 ,它主要負(fù)責(zé)處理和綁定到該handler的線程中的message。每一個handler都必須關(guān)聯(lián)一個looper,并且兩者是一一對應(yīng)的,注意,這點很重要哦!此外,looper負(fù)責(zé)從其內(nèi)部的messageQueue中拿出一個個的message給handler進(jìn)行處理。因為我們這里handler是在UI線程中實現(xiàn)的,所以經(jīng)過這么一個handler、message機制,我們就可以回到UI線程中了。

handler: 處理后臺進(jìn)程返回數(shù)據(jù)的工作人員。

message: 后臺進(jìn)程返回的數(shù)據(jù),里面可以存儲bundle等數(shù)據(jù)格式

messageQueue: 是線程對應(yīng)looper的一部分,負(fù)責(zé)存儲從后臺進(jìn)程中拋回的和當(dāng)前handler綁定的message,是一個隊列。

looper: looper相當(dāng)于一個messageQueue的管理人員,它會不停的循環(huán)的遍歷隊列,然后將符合條件的message一個個的拿出來交給handler進(jìn)行處理。

注意,handler是在UI線程中聲明的,如果我們直接用類似代碼執(zhí)行一個線程的話,實際上并沒有創(chuàng)建一個新的線程,因為handler已經(jīng)跟默認(rèn)的UI線程中的looper綁定了。

如果有興趣的話,可以去看下Handler的默認(rèn)空構(gòu)造函數(shù)便知道原因了,里面直接綁定了當(dāng)前UI線程的looper。

下面給出一個比較簡單,并且實用的實例。

    public class MainActivity extends Activity implements OnClickListener {
	private Button btnTXT;
	private TextView tvTXT;
	
	private StringBuffer returnMsg;
	
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        btnTXT = (Button)findViewById(R.id.btnTXT);
        tvTXT = (TextView)findViewById(R.id.tvTXT);
        
        btnTXT.setOnClickListener(this);		
    }

	@Override
	public void onClick(View v) {
		returnMsg = new StringBuffer();
		
		// 創(chuàng)建一個包含Looper的線程,這里如果沒有HandlerThread的調(diào)用,會直接將后邊的MyRunnable放到UI線程隊列(myHandler.post(new MyRunnable()))
		HandlerThread handlerThread = new HandlerThread("handler_thread");
		handlerThread.start();		// 啟動自定義處理線程
		
		myHandler = new MyHandler(handlerThread.getLooper());		// 將handler綁定到新線程  
		myHandler.post(new MyRunnable());		// 在新線程中執(zhí)行任務(wù) 
	}
    
	/** 主線程Handler,可以與UI控件交互 */
	Handler mainHanlder = new Handler(){
		@Override
		public void handleMessage(Message msg) {
			if(msg.what == 0) {
				tvTXT.setText(returnMsg.toString());	// 與主線程控件打交道(直接訪問)
			}
		}
	};

	
	/** 構(gòu)造Hanlder,不可與UI控件直接交互 */
	private MyHandler myHandler = null;
	private class MyHandler extends Handler{
		/** 
		 * 使用默認(rèn)的構(gòu)造函數(shù),會將handler綁定當(dāng)前UI線程的looper。 
		 * 如果想使用多線程這里是不能使用默認(rèn)的構(gòu)造方法的。 
		 */  
		public MyHandler(){
			super();
		}
		
		/** 構(gòu)造函數(shù),自定義looper */
		public MyHandler(Looper looper) {
			super(looper);
		}
		
		// 處理具體的message消息,繼承自父類的方法
		@Override
		public void handleMessage(Message msg) {
			int what = msg.what;	
			Bundle bundle = (Bundle)msg.obj;			// 提取bundle中的信息
			String name = bundle.getString("name");
			String sex = bundle.getString("sex");
			boolean marry = bundle.getBoolean("marray");
			int age = bundle.getInt("age");
			
			StringBuffer strBuf = new StringBuffer();		// 拼接bundle信息
			strBuf.append("what = ").append(what).append("\n\n");
			strBuf.append("name = ").append(name).append("\n");
			strBuf.append("sex = ").append(sex).append("\n");
			strBuf.append("marry = ").append(marry).append("\n");
			strBuf.append("age = ").append(age).append("\n\n");
			strBuf.append("http://blog.csdn.net/sunboy_2050");
			
			returnMsg = returnMsg.append(strBuf);	// 保存要顯示的結(jié)果
			mainHanlder.sendEmptyMessage(0);		// 向主線程mainHanlder發(fā)送消息,與UI控件交互顯示結(jié)果
			
			super.handleMessage(msg);
		}
	}
    
	
	// 構(gòu)造Runnable,處理后臺業(yè)務(wù)邏輯,如下載
	private class MyRunnable implements Runnable{
		@Override
		public void run() {
			try {
				Message msg = Message.obtain(myHandler);	// 捕獲myHandler消息
				
				msg.what = 10;
				Bundle bundle = new Bundle();				// 封裝bundle信息
				bundle.putString("name", "yanggang");
				bundle.putString("sex", "pure boy");
				bundle.putBoolean("marry", false);
				bundle.putInt("age", 18);
				msg.obj = bundle;
				
				long thID = Thread.currentThread().getId();
				returnMsg.append(thID).append(" : send msg start...").append("\n");
				msg.sendToTarget();		// 向myHandler發(fā)送消息

				Thread.sleep(3000);
			} catch (Exception e) {
				Log.i("", "Runnable send msg error...");
				e.printStackTrace();
			}
		}
	}
}
  

運行結(jié)果:

Android AsyncTask與handler


示例代碼下載



轉(zhuǎn)載聲明:

Android之多線程工作-AsyncTask與handler

Android自用-----AsyncTask實現(xiàn)異步處理任務(wù)

android線程 Handler Message Queue AsyncTask


參考推薦:

java synchronized詳解

java中synchronized用法

Android實現(xiàn)計時與倒計時的幾種方法

java同步synchronized關(guān)鍵字用法示例

Android AsyncTask與handler


更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯(lián)系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦!!!

發(fā)表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 91啦视频在线观看 | 全午夜免费一级毛片 | 免费 高清 日本1在线观看 | 国产欧美日韩成人 | 亚洲天堂h | 亚洲成人福利在线 | 99在线视频网站 | 久草在线视频免费播放 | 一区二区日韩 | 欧美亚洲日本国产综合网 | 亚洲美女精品视频 | 亚洲日日| 久久精品无遮挡一级毛片 | 久久在线免费视频 | 亚洲精品久久久中文字幕 | 欧美日本在线 | 天天干天操 | 又粗又大的机巴好爽7 | 日韩 亚洲 中文 图片 小说 | 欧美不卡一区 | 国产你懂的在线 | 97高清国语自产拍中国大陆 | 亚洲精品中文字幕字幕 | 91精品国产一区二区三区左线 | 97在线视频免费播放 | 亚洲一区二区中文字5566 | 日日操网| 成人欧美一区二区三区视频 | 六月婷婷久香在线视频 | 亚洲最大激情网 | 十八毛片 | 国产青草视频免费观看97 | 亚洲狠狠婷婷综合久久久久图片 | 国产综合色在线视频播放线视 | 国产99免费视频 | 老司机亚洲精品影视www | 99热这里都是国产精品 | 99爱色| 免费一级a毛片 | 天海翼一区二区在线观看 | 亚洲国产综合人成综合网站00 |