Android 主線程中 Looper.loop() 為什么不會(huì)卡死主線程魁兼?

第一次被問(wèn)到這個(gè)問(wèn)題的時(shí)候,就再想漠嵌,為什么會(huì)問(wèn)這問(wèn)題呢咐汞?
回想了一遍關(guān)于Android Handler,Message, MessageQueue 和 Looper 的相關(guān)知識(shí)盖呼,才明白為什么會(huì)有這樣的問(wèn)題。

這個(gè)問(wèn)題是怎么來(lái)的化撕?

因?yàn)?code>Looper.loop() 消息循環(huán)是個(gè)死循環(huán)几晤,會(huì)不斷的在這里處理MessageQueue消息隊(duì)列中的消息

 /**
     * Run the message queue in this thread. Be sure to call
     * {@link #quit()} to end the loop.
     */
    public static void loop() {
        final Looper me = myLooper();
        if (me == null) {
            throw new RuntimeException("No Looper; Looper.prepare() wasn't called on this thread.");
        }
        final MessageQueue queue = me.mQueue;
        /*
         * 刪除了一些無(wú)關(guān)代碼
         */
        for (;;) {
            // 這里循環(huán)從queue中獲取Message消息,如果沒(méi)有消息的話這里會(huì)阻塞
            Message msg = queue.next(); // might block
            if (msg == null) {
                // No message indicates that the message queue is quitting.
                return;
            }

            try {
                //msg.target 對(duì)象是發(fā)送Message消息的Handler對(duì)象植阴,通過(guò)Handler的dispatchMessage進(jìn)行消息處理
                msg.target.dispatchMessage(msg);
                dispatchEnd = needEndTime ? SystemClock.uptimeMillis() : 0;
            } finally {
                if (traceTag != 0) {
                    Trace.traceEnd(traceTag);
                }
            }
             /*
              * 刪除了一些無(wú)關(guān)代碼
              */
            // 消息處理完后锌仅,將Message放入對(duì)象池中,這樣Message.obtain()獲取Message時(shí)候可以減少對(duì)象的創(chuàng)建
            msg.recycleUnchecked();
        }
    }

Looper.loop() 為什么不會(huì)卡死主線程

1, 這里涉及到Linux pipe/epoll機(jī)制 ,簡(jiǎn)單說(shuō)就是在主線程的MessageQueue沒(méi)有消息時(shí)墙贱,便阻塞在loop的queue.next()中的nativePollOnce()方法里热芹。此時(shí)主線程會(huì)釋放CPU資源進(jìn)入休眠狀態(tài),直到下個(gè)消息到達(dá)或者有事務(wù)發(fā)生惨撇,通過(guò)往pipe管道寫端寫入數(shù)據(jù)來(lái)喚醒主線程工作伊脓。這里采用的epoll機(jī)制,是一種IO多路復(fù)用機(jī)制魁衙,可以同時(shí)監(jiān)控多個(gè)描述符报腔,當(dāng)某個(gè)描述符就緒(讀或?qū)懢途w),則立刻通知相應(yīng)程序進(jìn)行讀或?qū)懖僮髌实恚举|(zhì)同步I/O纯蛾,即讀寫是阻塞的。

2,在進(jìn)入死循環(huán)之前創(chuàng)建了新binder線程纵隔,在代碼ActivityThread.main()中,
thread.attach(false)翻诉;便會(huì)創(chuàng)建一個(gè)Binder線程(具體是指ApplicationThread,Binder的服務(wù)端捌刮,用于接收系統(tǒng)服務(wù)AMS發(fā)送來(lái)的事件)碰煌,該Binder線程通過(guò)Handler將Message發(fā)送給主線程.

public static void main(String[] args) {
        ....

        //創(chuàng)建Looper和MessageQueue對(duì)象,用于處理主線程的消息
        Looper.prepareMainLooper();

        //創(chuàng)建ActivityThread對(duì)象
        ActivityThread thread = new ActivityThread(); 

        //建立Binder通道 (創(chuàng)建新線程)
        thread.attach(false);

        Looper.loop(); //消息循環(huán)運(yùn)行
        throw new RuntimeException("Main thread loop unexpectedly exited");
    }

所以 在 Looper.loop() 中阻塞的時(shí)候不會(huì)卡死主線程绅作。

Looper.loop() 死循環(huán)會(huì)特別消耗CPU資源嗎芦圾?

這里同樣是涉及到Linux pipe/epoll機(jī)制,可參考不卡死主線程的原因俄认。對(duì)Linux pipe/epoll機(jī)制有興趣可google相關(guān)的介紹

Looper.loop() 會(huì)發(fā)生ANR 嗎个少?

下面是ANR在官方文檔的介紹:

ANR

如果 Android 應(yīng)用的界面線程處于阻塞狀態(tài)的時(shí)間過(guò)長(zhǎng),會(huì)觸發(fā)“應(yīng)用無(wú)響應(yīng)”(ANR) 錯(cuò)誤眯杏。如果應(yīng)用位于前臺(tái)夜焦,系統(tǒng)會(huì)向用戶顯示一個(gè)對(duì)話框。ANR 對(duì)話框會(huì)為用戶提供強(qiáng)行退出應(yīng)用的選項(xiàng)役拴。

可以看到ANR的發(fā)生是在程序在處理Message消息的時(shí)候糊探,用的時(shí)間太長(zhǎng)钾埂,導(dǎo)致 Looper.loop() 無(wú)法進(jìn)入下一個(gè)循環(huán)處理后續(xù)的消息河闰。

所以 ANR 和 Looper.loop()的阻塞 是兩個(gè)不同的概念科平。

  • Looper.loop()阻塞是消息隊(duì)列為空,在等待新的消息姜性,然后進(jìn)行處理瞪慧。
  • ANR 是消息隊(duì)列不為空的時(shí)候,程序在處理某一次的Message時(shí)部念,系統(tǒng)檢測(cè)耗時(shí)太久弃酌,提示的ANR。

END儡炼!

參考:
https://www.zhihu.com/question/34652589
https://developer.android.com/topic/performance/vitals/anr

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末妓湘,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子乌询,更是在濱河造成了極大的恐慌榜贴,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,744評(píng)論 6 502
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件妹田,死亡現(xiàn)場(chǎng)離奇詭異唬党,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)鬼佣,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,505評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門驶拱,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人晶衷,你說(shuō)我怎么就攤上這事蓝纲。” “怎么了晌纫?”我有些...
    開封第一講書人閱讀 163,105評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵驻龟,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我缸匪,道長(zhǎng)翁狐,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,242評(píng)論 1 292
  • 正文 為了忘掉前任凌蔬,我火速辦了婚禮露懒,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘砂心。我一直安慰自己懈词,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,269評(píng)論 6 389
  • 文/花漫 我一把揭開白布辩诞。 她就那樣靜靜地躺著坎弯,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上抠忘,一...
    開封第一講書人閱讀 51,215評(píng)論 1 299
  • 那天撩炊,我揣著相機(jī)與錄音,去河邊找鬼崎脉。 笑死拧咳,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的囚灼。 我是一名探鬼主播骆膝,決...
    沈念sama閱讀 40,096評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼灶体!你這毒婦竟也來(lái)了阅签?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,939評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤蝎抽,失蹤者是張志新(化名)和其女友劉穎愉择,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體织中,經(jīng)...
    沈念sama閱讀 45,354評(píng)論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡锥涕,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,573評(píng)論 2 333
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了狭吼。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片层坠。...
    茶點(diǎn)故事閱讀 39,745評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖刁笙,靈堂內(nèi)的尸體忽然破棺而出破花,到底是詐尸還是另有隱情,我是刑警寧澤疲吸,帶...
    沈念sama閱讀 35,448評(píng)論 5 344
  • 正文 年R本政府宣布座每,位于F島的核電站,受9級(jí)特大地震影響摘悴,放射性物質(zhì)發(fā)生泄漏峭梳。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,048評(píng)論 3 327
  • 文/蒙蒙 一蹂喻、第九天 我趴在偏房一處隱蔽的房頂上張望葱椭。 院中可真熱鬧,春花似錦口四、人聲如沸孵运。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,683評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)治笨。三九已至驳概,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間旷赖,已是汗流浹背顺又。 一陣腳步聲響...
    開封第一講書人閱讀 32,838評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留杠愧,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,776評(píng)論 2 369
  • 正文 我出身青樓逞壁,卻偏偏與公主長(zhǎng)得像流济,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子腌闯,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,652評(píng)論 2 354

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