理解Handler的原理首先要搞清楚什么是Looper,在我的上一篇博文中對(duì)此有專門的介紹。Looper的作用是開啟一個(gè)消息循環(huán),從MessageQueue(Message隊(duì)列,是Looper的成員變量)中循環(huán)取出消息處理。一個(gè)線程要使用Handler來(lái)處理來(lái)自其它線程的消息,這個(gè)線程必須有且僅有一個(gè)Looper對(duì)象與之綁定,也可以說(shuō)一個(gè)Looper對(duì)象是是與一個(gè)線程一一對(duì)應(yīng)的。
10年積累的網(wǎng)站建設(shè)、網(wǎng)站設(shè)計(jì)經(jīng)驗(yàn),可以快速應(yīng)對(duì)客戶對(duì)網(wǎng)站的新想法和需求。提供各種問(wèn)題對(duì)應(yīng)的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務(wù)。我雖然不認(rèn)識(shí)你,你也不認(rèn)識(shí)我。但先網(wǎng)站設(shè)計(jì)后付款的網(wǎng)站建設(shè)流程,更有鹽邊免費(fèi)網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。
Hander有一個(gè)Looper類型的成員,在Handler的構(gòu)造函數(shù)(new Handler()或者new Handler(CallBack))中會(huì)實(shí)例化這個(gè)Handler的Looper成員,Handler()構(gòu)造函數(shù)的源碼如下:
public Handler() { //獲得當(dāng)前線程在 ThreadLocal 中所對(duì)應(yīng)的 Looper mLooper = Looper.myLooper(); if (mLooper == null) { throw new RuntimeException( "Can't create handler inside thread that has not called Looper.prepare()" ); } mQueue = mLooper.mQueue; mCallback = null; }
在Handler構(gòu)造函數(shù)中實(shí)例化的Looper不是通過(guò)new關(guān)鍵字來(lái)實(shí)例化的,而是從Looper.myLooper()這個(gè)靜態(tài)方法中取得的。而Looper.myLooper()又是從哪來(lái)取得的Looper對(duì)象呢?這就涉及到另外一個(gè)類ThreadLocal。Looper有一個(gè)靜態(tài)變量是ThreadLocal類型的:
static final ThreadLocal
這個(gè)變量就是保存每一個(gè)線程和這個(gè)線程對(duì)應(yīng)的Looper。這樣設(shè)計(jì)的作用是保證每一個(gè)線程的Looper是唯一的。
每一個(gè)線程在new Handler()之前必須為這個(gè)線程創(chuàng)建Looper對(duì)象,使用Looper.prepare()方法創(chuàng)建。
Looper.prepare的源碼如下:
public static void prepare() { //調(diào)用此方法的線程是否在 全局變量 sThreadLocal(即Map)中存有一組以此線程為鍵的鍵值對(duì) if (sThreadLocal.get() != null) { throw new RuntimeException("Only one Looper may be created per thread"); } //如果在Map 中沒(méi)有存放當(dāng)前線程對(duì)應(yīng)的鍵值對(duì)當(dāng)存入 // 一個(gè) set 操作,其實(shí)就是在 Map 中插入了一組 值,即 <調(diào)用此方法的線程,new Looper()> // new Looper() 時(shí),Looper的構(gòu)造函數(shù)就會(huì)實(shí)例化一個(gè) MessageQueue sThreadLocal.set(new Looper());
關(guān)鍵是sThreadLocal.set(new Looper());這條語(yǔ)句,它會(huì)把new 出來(lái)的Looper保存到ThreadLocal這個(gè)全局變量中。
有一個(gè)問(wèn)題,就是為什么我們?cè)谥骶€程中new Handler(){...}之前不需要使用Looper.prepare()呢?因?yàn)樵谥骶€程執(zhí)行之前,android虛擬機(jī)已經(jīng)幫我們執(zhí)行了這段代碼,因此在主線程中創(chuàng)建Handler對(duì)象不需要再新建Looper對(duì)象。創(chuàng)建了Handler之后,還要使用Looper.loop()開啟消息循環(huán)來(lái)取消息。主線程也不需要這句代碼。
總結(jié):
Looper負(fù)責(zé)開啟消息循環(huán),從MessageQueue中讀取Message,由Handler負(fù)責(zé)處理讀出來(lái)的Message。
Looper和它的MessageQueue與某一個(gè)線程是一一對(duì)應(yīng)的。
使用Handler之前需要使用Looper.prepare()為當(dāng)前線程創(chuàng)建Looper對(duì)象。
使用Looper.loop()開啟消息循環(huán)。