30分鐘帶你搞懂Handler

Handler是什么阻肩?

對(duì)于Handler 我們都不太陌生 這是我們Android的消息傳輸機(jī)制 我們都知道它是用來(lái)幫我們通過(guò)子線程發(fā)送消息到主線程 供我們主線程進(jìn)行更新UI操作的 那他為什么可以完成線程間的通信呢污筷?

Handler 包含了什么审编?

Handler更新UI界面的機(jī)制厘熟,也是消息處理的機(jī)制,可以發(fā)送消息,也可以處理消息
Message 接收處理消息對(duì)象
Looper 每個(gè)線程只有1個(gè):可讀取MessageQueue消息隊(duì)列时呀,管理此線程里的MessageQueue(消息隊(duì)列)遵循先進(jìn)先出讀到消息后從消息隊(duì)列取出交給handler
Message Queue(消息隊(duì)列):用來(lái)存放線程放入的消息

Handler消息機(jī)制是什么芦疏?

Handler 機(jī)制中包含了幾個(gè)關(guān)鍵的對(duì)象,Looper,MessageQueue,Message矛洞。在使用 Handler 的時(shí)候葬燎,在 handler 所創(chuàng)建的線程需要維護(hù)一個(gè)唯一的 Looper 對(duì)象,每個(gè)線程對(duì)應(yīng)一個(gè) Looper, 每個(gè)線程的 Looper 通過(guò) ThreadLocal 來(lái)保證缚甩。Looper 對(duì)象的內(nèi)部又維護(hù)有唯一的一個(gè)MessageQueue 鏈表,所以一個(gè)線程可以有多個(gè) Handler,但是只能有一個(gè) Looper 和一個(gè)MessageQueue窑邦。Message 在 MessageQueue 不是通過(guò)一個(gè)列表來(lái)存儲(chǔ)的擅威,而是將傳入的Message 存入到了上一個(gè) Message 的 next 中,在取出的時(shí)候通過(guò)頂部的 Message 按放入的順序依次取出 Message冈钦。Looper 對(duì)象通過(guò) loop()方法開(kāi)啟了一個(gè)死循環(huán)郊丛,不斷地從 looper 內(nèi)的 MessageQueue 中取出 Message,然后通過(guò) handler 將消息分發(fā)傳回 handler 所在的線程

如果對(duì)Handler有了解的朋友會(huì)覺(jué)得卻是就是挺簡(jiǎn)單的 但是對(duì)于初學(xué)者來(lái)說(shuō) 這TM是啥都是些什么玩意 哈哈哈 不要著急 我們今天通過(guò)源碼來(lái)給大家講解一下 關(guān)于Handler機(jī)制的執(zhí)行流程

我們先來(lái)通過(guò)一張流程圖來(lái)看一下我們是怎么把消息從子線程發(fā)送到主線程的:


image.png

Handler源碼

首先我們都知道Activity創(chuàng)建以后會(huì)首先走到我們ActivityTrhead 的main 函數(shù)中(也就是我們的main方法)

ActivityThread.png

這里我們發(fā)現(xiàn) 咦 怎么我們main方法中還會(huì)幫我們創(chuàng)建Looper的? 當(dāng)你對(duì)java夠熟悉的時(shí)候 大家就可以進(jìn)入源碼中了解一下 我們就會(huì)發(fā)現(xiàn) 其實(shí)我們對(duì)代碼的理解過(guò)于偏淺 咳咳 跑題了 我們首先去我們的第一步Looper.prepareMainLooper()這個(gè)方法中看一看:


image.png

我們發(fā)現(xiàn)其實(shí)就是調(diào)用了Looper中的一個(gè)方法 進(jìn)入方法第一步就執(zhí)行了一個(gè) 本類(lèi)中的prepare()方法 這個(gè)方法是做什么的呢瞧筛? 我們點(diǎn)進(jìn)去繼續(xù)看:


image.png

這里我們發(fā)現(xiàn)它給我們做了一個(gè)判斷 sTreadLocal.get() 獲取到的這個(gè)對(duì)象是否不為空 不為空就給我們拋出異常 否則就幫我們set()一個(gè)厉熟。 我們這個(gè)時(shí)候就想問(wèn)了 這個(gè)sTreadLocal是個(gè)什么東西?為什么要通過(guò)他來(lái)獲取Looper呢较幌?

image.png

我們發(fā)現(xiàn)這個(gè)sThreadLocal 就是線程內(nèi)部的數(shù)據(jù)存儲(chǔ)類(lèi)的對(duì)象 揍瑟,通過(guò)他可以在指定的線程中存儲(chǔ)數(shù)據(jù),該數(shù)據(jù)只有在指定線程中可以獲取
我們看一下這個(gè)ThreadLocal給我們提供的set() get()方法:

set()方法:


image.png

這里我們發(fā)現(xiàn)一個(gè)ThreadLocalMap 一個(gè)map集合 我們發(fā)現(xiàn)這一步吧我們的當(dāng)前線程和我們當(dāng)前線程的Looper對(duì)象 通過(guò) key value 的形式 綁定到了一個(gè)ThreadLocalMap 對(duì)象中
我們看一下Looper的構(gòu)造方法中都創(chuàng)建了什么:

image.png

我們發(fā)現(xiàn)它幫我們創(chuàng)建了一個(gè)MessageQueue();并把我們的的當(dāng)前線程對(duì)象賦值給了mThread 這里特別重要 我們的Looper只能有一個(gè)MessageQueue 而且是在創(chuàng)建Looper時(shí)創(chuàng)建了 也就是同步創(chuàng)建的乍炉。

有了賦值set() 我們?cè)賮?lái)看看取值get()方法

get()方法 :


image.png

image.png

image.png

get()方法中代碼很多 但是原理很簡(jiǎn)單 就是有就使用 沒(méi)有就賦值嘛
我們只需要知道我們通過(guò)這個(gè)get()方法返回的是我們當(dāng)前線程的Looper對(duì)象就可以啦

弄懂這兩個(gè)方法以后我們就知道了我們prepare這個(gè)方法中都了什么 :判斷我們Looper是否存在 不存在的話就幫我們創(chuàng)建 存在就從ThreadLocalMap中取出當(dāng)前線程對(duì)應(yīng)的Looper對(duì)象

我們繼續(xù)往下看:
prepareMainLooper這個(gè)方法:


image.png

我們這里的sMainLooper就是一個(gè)默認(rèn)為空的Loopr對(duì)象


image.png

我們看到如果他存在的話 就拋出異常 不存在的話就調(diào)用本類(lèi)的myLooper方法
這個(gè)myLooper都干了什么呢绢片?我們點(diǎn)進(jìn)去看一下 :


image.png

這個(gè)地方就通過(guò)我們ThreadLocal的get()方法取出我們當(dāng)前線程的Looper對(duì)象

到這里我們的 Looper.prepareMainLooper(); 創(chuàng)建主線程Looper對(duì)象就完成了

我們接下來(lái)往下看


image.png

我們發(fā)現(xiàn)我們這里又調(diào)用了Looper.loop() 這個(gè)方法滤馍, 這個(gè)方法就是我們輪詢(xún)消息的方法


image.png

又是一段長(zhǎng)代碼 是不是看的讓人頭大 不要慌 看源碼就是很繁瑣的 我們要有耐心
我們首先看到 調(diào)用了myLooper()這個(gè)方法 這個(gè)方法是做什么呢?

image.png

我們發(fā)現(xiàn)它和我們ThreadLocal.get()這個(gè)方法功能是一樣的 都是獲取到我們保存的Looper對(duì)象
我們接下來(lái)看它又做了哪些事呢底循?
image.png

這里把我們Looper對(duì)象的MessageQueue取出來(lái)
image.png

通過(guò)死循環(huán) 不斷的調(diào)用MessageQueue 消息隊(duì)列 對(duì)象的next()方法 獲取到Message 直到我們的消息隊(duì)列中沒(méi)有消息 返回 這個(gè)next()方法里面是怎么把消息給我們?nèi)〕瞿兀?br> 這個(gè)方法中代碼太多 我就把核心給取出展示:
image.png

image.png

這里就是定義一個(gè)空的Message 然后把我們的消息隊(duì)列中的消息 進(jìn)行取出 把每一個(gè)值都賦值給我們空的Message 然后把我們賦值過(guò)后的Message返回

如果我們的消息隊(duì)列中有消息的話 就會(huì)一直調(diào)用這個(gè)next方法進(jìn)行取值 直到消息隊(duì)列中的消息為空 next就會(huì)堵塞線程 否則loop循環(huán)就會(huì)無(wú)限循環(huán)下去 當(dāng)我們獲取到消息以后我們用做了什么事情呢巢株?

image.png

我們發(fā)現(xiàn)我們調(diào)用了一個(gè)msg.target.dispatchMessage()方法 首先我們要知道m(xù)sg.target是什么


image.png

原來(lái)他是一個(gè)Handler 的對(duì)象 我們接下來(lái)再看dispatchMessage這個(gè)是干了什么:


image.png

我*** 有沒(méi)有感覺(jué)到最下面那一行代碼特別熟悉:
這就是我們創(chuàng)建Handler的時(shí)候


image.png

這個(gè)handleMessage方法嗎 原來(lái)就是這個(gè)地方把我們的msg傳遞給我們的Handler了啊 是不是覺(jué)得其實(shí)也不是那么的困難 看完了以上的代碼 我們接下來(lái)再來(lái)看我們的Handler 哈哈哈哈哈 這里才到我們的Handler 是不是覺(jué)得已經(jīng)很無(wú)聊了 馬上步入正題:

我們先看一下我們創(chuàng)建Handler的時(shí)候都幫我們干了什么:


image.png

繼續(xù)往里看:


image.png

我們發(fā)現(xiàn)它調(diào)用了myLooper()這個(gè)方法 這個(gè)發(fā)法我們之前已經(jīng)看到過(guò)了:
是用來(lái)獲取我們當(dāng)前線程的Looper對(duì)象的 接下來(lái)又把我們當(dāng)前線程Looper對(duì)象中的MessageQueue綁定到我們的Handler中的MessageQueue 。接下來(lái)就來(lái)看我們的發(fā)送消息事件吧
image.png

我們今天通過(guò)一個(gè)簡(jiǎn)單的發(fā)送消息來(lái)進(jìn)行講解:
我們點(diǎn)進(jìn)去sendMessage方法發(fā)現(xiàn):


image.png

這里發(fā)送了一條默認(rèn)為0的延遲消息 這個(gè)方法又做了什么呢:

image.png

又指定時(shí)間發(fā)送了一條消息-->

image.png

這個(gè)方法定義了一個(gè)空的MessageQueue綁定我們Handler的MessageQueue 又往下走enqueueMessage()方法:

image.png

我們發(fā)現(xiàn)這里把我們存儲(chǔ)到msg.target的這個(gè)Handler賦值給了我們當(dāng)前的Handler 接下來(lái)調(diào)用了MessageQueue的enqueueMessage() :


image.png

首先對(duì)我們的 msg.target 也就是 Hander進(jìn)行了一次判空 如果為空就給我們拋出異常

         Message p = mMessages;
            boolean needWake;
            if (p == null || when == 0 || when < p.when) {
                // New head, wake up the event queue if blocked.
                msg.next = p;
                mMessages = msg;
                needWake = mBlocked;
            } else {
                // Inserted within the middle of the queue.  Usually we don't have to wake
                // up the event queue unless there is a barrier at the head of the queue
                // and the message is the earliest asynchronous message in the queue.
                needWake = mBlocked && p.target == null && msg.isAsynchronous();
                Message prev;
                for (;;) {
                    prev = p;
                    p = p.next;
                    if (p == null || when < p.when) {
                        break;
                    }
                    if (needWake && p.isAsynchronous()) {
                        needWake = false;
                    }
                }
                msg.next = p; // invariant: p == prev.next
                prev.next = msg;
            }

首先判斷我們消息隊(duì)列中是否有消息 如果沒(méi)有就直接放到第一個(gè) 如果有消息的話就放到他的后面

這個(gè)時(shí)候我們的消息就完全的存放到了我們MessageQueue中 我們的的ActivityThread 的main 方法中的Looper.loop 就會(huì)把我們消息隊(duì)列中的消息輪詢(xún)?nèi)〕?給我們handler使用了

還有一點(diǎn)就是我們Handler的內(nèi)存泄漏問(wèn)題:

在使用 Handler 的過(guò)程中熙涤,需要注意內(nèi)存泄漏阁苞,因?yàn)?handler 是用來(lái)進(jìn)行線程間通信的,所
以新開(kāi)啟的線程會(huì)持有 Handler 引用的祠挫,如果 Activity 中創(chuàng)建 Handler那槽,并且是非靜態(tài)內(nèi)部類(lèi)
的形式,就有可能造成內(nèi)存泄漏茸歧。非靜態(tài)的內(nèi)部類(lèi)是會(huì)隱式持有外部類(lèi)的引用倦炒,所以當(dāng)其他
線程持有了 handler,線程沒(méi)有被銷(xiāo)毀,就意味著 Activity 會(huì)一致被 Handler 持有引用而無(wú)法導(dǎo)
致回收软瞎。同時(shí) MessageQueue 中如果存在未處理完的 Message,Message 的 target 也是對(duì) Acivity
的持有引用逢唤,也會(huì)造成內(nèi)存泄漏。
解決的方法涤浇,可以使用靜態(tài)的內(nèi)部類(lèi)+弱引用的方式鳖藕。在外部類(lèi)對(duì)象被銷(xiāo)毀時(shí),將
MessageQueue 中 的 消 息 清 空 只锭。 如 Activity 的 onDestroy 時(shí) 將 消 息 情 況 著恩, 調(diào) 用
handler.removeCallbacksAndMessages(null); 清空消息

以上就是我總結(jié)的Handler啦 有沒(méi)有學(xué)到什么呢~
如果您發(fā)現(xiàn)了哪些錯(cuò)誤可以聯(lián)系我哦~

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市蜻展,隨后出現(xiàn)的幾起案子喉誊,更是在濱河造成了極大的恐慌,老刑警劉巖纵顾,帶你破解...
    沈念sama閱讀 222,807評(píng)論 6 518
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件伍茄,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡施逾,警方通過(guò)查閱死者的電腦和手機(jī)敷矫,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,284評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)汉额,“玉大人曹仗,你說(shuō)我怎么就攤上這事∪渌眩” “怎么了怎茫?”我有些...
    開(kāi)封第一講書(shū)人閱讀 169,589評(píng)論 0 363
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)妓灌。 經(jīng)常有香客問(wèn)我遭居,道長(zhǎng)啼器,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 60,188評(píng)論 1 300
  • 正文 為了忘掉前任俱萍,我火速辦了婚禮端壳,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘枪蘑。我一直安慰自己损谦,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,185評(píng)論 6 398
  • 文/花漫 我一把揭開(kāi)白布岳颇。 她就那樣靜靜地躺著照捡,像睡著了一般。 火紅的嫁衣襯著肌膚如雪话侧。 梳的紋絲不亂的頭發(fā)上栗精,一...
    開(kāi)封第一講書(shū)人閱讀 52,785評(píng)論 1 314
  • 那天,我揣著相機(jī)與錄音瞻鹏,去河邊找鬼悲立。 笑死,一個(gè)胖子當(dāng)著我的面吹牛新博,可吹牛的內(nèi)容都是我干的薪夕。 我是一名探鬼主播,決...
    沈念sama閱讀 41,220評(píng)論 3 423
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼赫悄,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼原献!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起埂淮,我...
    開(kāi)封第一講書(shū)人閱讀 40,167評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤姑隅,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后倔撞,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體粤策,經(jīng)...
    沈念sama閱讀 46,698評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,767評(píng)論 3 343
  • 正文 我和宋清朗相戀三年误窖,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片秩贰。...
    茶點(diǎn)故事閱讀 40,912評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡霹俺,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出毒费,到底是詐尸還是另有隱情丙唧,我是刑警寧澤,帶...
    沈念sama閱讀 36,572評(píng)論 5 351
  • 正文 年R本政府宣布觅玻,位于F島的核電站想际,受9級(jí)特大地震影響培漏,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜胡本,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,254評(píng)論 3 336
  • 文/蒙蒙 一牌柄、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧侧甫,春花似錦珊佣、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,746評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至守屉,卻和暖如春惑艇,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背拇泛。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,859評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工滨巴, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人碰镜。 一個(gè)月前我還...
    沈念sama閱讀 49,359評(píng)論 3 379
  • 正文 我出身青樓兢卵,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親绪颖。 傳聞我的和親對(duì)象是個(gè)殘疾皇子秽荤,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,922評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容