Hanlder源碼分析(包括同步屏障,異步消息)

1 創(chuàng)建Handler

眾所周知,在子線程直接創(chuàng)建Handler一定會報錯,如圖


1.png

意思也很明確,必須要調(diào)用Looper.prepare(),才能創(chuàng)建Handler,因為整個Handler的消息循環(huán)機制是建立在Looper之上.

那為什么主線程不用調(diào)用Looper.prepare就可以創(chuàng)建Handler呢?
因為在ActivityThread的main函數(shù)里面,幫我們調(diào)用了Looper.prepare()

public static void main(String[] args) {
        //省略部分代碼
        ...
        Looper.prepareMainLooper();

        ActivityThread thread = new ActivityThread();
        thread.attach(false);

        if (sMainThreadHandler == null) {
            sMainThreadHandler = thread.getHandler();
        }

        if (false) {
            Looper.myLooper().setMessageLogging(new
                    LogPrinter(Log.DEBUG, "ActivityThread"));
        }

        // End of event ActivityThreadMain.
        Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
        Looper.loop();

        throw new RuntimeException("Main thread loop unexpectedly exited");
    }

而使用兩次Looper.prepare()則會報一下錯誤


2.png

因為每個Thread僅僅只能獲取一個Looper.

搞清楚了Looper.prepare,我們來看一下為什么必須要調(diào)用它才能創(chuàng)建Handler.

2 Looper.prepare

3.png

4.png

5.png

6.png

而Looper的mQueue則是在構(gòu)造函數(shù)中創(chuàng)建的,如圖所示,即我們調(diào)用prepare的時候
到此,prepare的代碼全部執(zhí)行完了。

從上面的圖6,可以看到prepare主要里是這句代碼
sThreadLocal.set(new Looper(quitAllowed));
1 新建了一個Loopper對象
2 set到sThreadLocal里面
3 sThreadLocal又新建了一個Map來保存當(dāng)前的key(即當(dāng)前的Thread)與value(即新建的Looper)
4 Looper創(chuàng)建了MessageQueue

再來看new Handler做了什么事情

3 Handler構(gòu)造函數(shù)

7.png

可以看到這句 mLooper = Looper.myLooper();關(guān)鍵代碼,將Looper.myLooper賦值給Handler自己的mLooper,不然就報那個知名的錯誤枫慷。

mQueue = mLooper.mQueue;//將looper的mQueue賦值給Handler的mQueue

8.png

image.png

可以看到Looper.myLooper()是將prepare的里面創(chuàng)建的Looper再取出來巴比。
可以得知 每次新建Handler的時候,得先將Looper創(chuàng)建好。

4 Looper.loop

一般我們創(chuàng)建好Handler之后,都需要在調(diào)用Looper.loop,我們看看它具體做了什么事情.

public static void loop() {
        final Looper me = myLooper();
        //省略部分無效代碼
        final MessageQueue queue = me.mQueue;

        for (;;) {
            Message msg = queue.next(); // might block
            if (msg == null) {
                // No message indicates that the message queue is quitting.
                return;
            }
          try {
                msg.target.dispatchMessage(msg);
                end = (slowDispatchThresholdMs == 0) ? 0 : SystemClock.uptimeMillis();
            } finally {
                if (traceTag != 0) {
                    Trace.traceEnd(traceTag);
                }
            }
            //省略部分無效代碼
            msg.recycleUnchecked();
        }
    }

至此,我們可以看到,loop是獲取了,MessageQueue,然后啟動死循環(huán),不斷調(diào)用queue.next直到消息隊列里,沒有任何消息了,才跳出循環(huán).
至此,創(chuàng)建Handler的部分結(jié)束了,我們來看看Handler如何SendMessage的

5 Handler.SendMessage

當(dāng)我們調(diào)用SendMessage時候,最終會調(diào)用的方法是這個


9.png

10.png
boolean enqueueMessage(Message msg, long when) {
        if (msg.target == null) {
            throw new IllegalArgumentException("Message must have a target.");
        }
        if (msg.isInUse()) {
            throw new IllegalStateException(msg + " This message is already in use.");
        }

        synchronized (this) {

            msg.markInUse();
            msg.when = when;
            Message p = mMessages;
            boolean needWake;
            if (p == null || when == 0 || when < p.when) {// 如果是第一個進入到消息隊列的消息且delay的時間最小(最小為0),則把其設(shè)置為消息隊列的頭
                // New head, wake up the event queue if blocked.
                msg.next = p;
                mMessages = msg;
                needWake = mBlocked;
            } else {
            //鏈表操作,增加一個消息實體
                Message prev;
                for (;;) {
                    prev = p;
                    p = p.next;
                    if (p == null || when < p.when) {//根據(jù)delay的時間決定插在隊列的什么位置
                        break;
                    }
                }
                msg.next = p; // invariant: p == prev.next
                prev.next = msg;
            }
        }
        return true;
    }

可以看到,MessageQueue.enqueueMessage會將所有的被sendMessage發(fā)送的Message增加到鏈表里登夫。
由于Looper.loop的里面調(diào)用queue.next()的關(guān)系,(一直查詢是否有新消息到來),如果查詢到新消息,則進行msg.target.dispatchMessage(msg)。而msg的target則是在enqueueMessage做msg.target = this了,既當(dāng)前發(fā)送此消息的Handler.
此時又回到了Handler的dispatchMessage

6 Handler.dispatchMessage

11.png

可以看到,dispatchMessage并沒有做什么復(fù)雜的操作,僅僅就是判斷了Message自身是否有CallBack,Handler是否設(shè)置了CallBack,如果都沒有,則調(diào)用Handler的HandleMessage方法。

同步屏障

ViewRootImpl.java


 void scheduleTraversals() {
        if (!mTraversalScheduled) {
            mTraversalScheduled = true;
            mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();//建立同步屏障
            mChoreographer.postCallback(
                    Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null); //發(fā)送消息
            if (!mUnbufferedInputDispatch) {
                scheduleConsumeBatchedInput();
            }
            notifyRendererOfFramePending();
            pokeDrawLockIfNeeded();
        }
    }

    void unscheduleTraversals() {
        if (mTraversalScheduled) {
            mTraversalScheduled = false;
            mHandler.getLooper().getQueue().removeSyncBarrier(mTraversalBarrier);
            mChoreographer.removeCallbacks(
                    Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
        }
    }

Choreographer.postCallback實際上調(diào)用了postCallbackDelayedInternal,可以看到setAsynchronous(true);即設(shè)置為異步消息狈茉。

private void postCallbackDelayedInternal(int callbackType,
            Object action, Object token, long delayMillis) {
        if (DEBUG_FRAMES) {
            Log.d(TAG, "PostCallback: type=" + callbackType
                    + ", action=" + action + ", token=" + token
                    + ", delayMillis=" + delayMillis);
        }

        synchronized (mLock) {
            final long now = SystemClock.uptimeMillis();
            final long dueTime = now + delayMillis;
            mCallbackQueues[callbackType].addCallbackLocked(dueTime, action, token);

            if (dueTime <= now) {
                scheduleFrameLocked(now);
            } else {
                Message msg = mHandler.obtainMessage(MSG_DO_SCHEDULE_CALLBACK, action);
                msg.arg1 = callbackType;
                msg.setAsynchronous(true);//設(shè)置為異步消息
                mHandler.sendMessageAtTime(msg, dueTime);
            }
        }
    }

總結(jié):
1 調(diào)用Looper.prepare是為了創(chuàng)建Looper,然后創(chuàng)建Map,根據(jù)CurrentThread保存Looper。Looper創(chuàng)建了MessageQueue
2 new Handler的時候會將Looper與Hanlder綁定,并且將Looper的MessageQueue與Handler的mQueue綁定
3 Looper.loop是為了啟動消息循環(huán)不斷查詢(queue.next())是否有新消息到來,可能會阻塞
4 Handler.SendMessage則是將Message.target與Handler綁定,并且將Message插入到MessageQueue中去(根據(jù)delay插入,越小越前)
5 Looper.loop中查詢到有新消息來了后,將會調(diào)用Message.target.dispatchMessage,將msg分發(fā)出去
6 Hanlder.dispatchMessage將進行msg的callback,Hanlder的callback判斷,最后才調(diào)用HandleMessage.
即Message的CallBack優(yōu)先級最高,Hanlder的CallBack其次,HandleMessage最低.
7 sThreadLocal是確保每個Thread里有且僅有一個Looper的關(guān)鍵,因為會去獲取線程獨有的ThreadLocal.ThreadLocalMap作為Map,然后把自己作為key,來保存looper,每次prepare則先會訪問線程獨有的ThreadLocal.ThreadLocalMap來判斷是否有value
8 Message也分同步消息與異步消息,有兩種方式發(fā)送異步消息掸掸, Message.setAsynchronous(True)與Hanlder.createAsync(),如果不設(shè)置則都為同步消息氯庆。在系統(tǒng)的ViewRootImpl里面Chreograher就是發(fā)送異步消息來繪制UI 即perfromTraversal會提前于所有同步消息執(zhí)行
9 延時消息 是利用Message的when來對比,插入到哪里(小的前,大的后)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市扰付,隨后出現(xiàn)的幾起案子堤撵,更是在濱河造成了極大的恐慌,老刑警劉巖羽莺,帶你破解...
    沈念sama閱讀 217,734評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件实昨,死亡現(xiàn)場離奇詭異,居然都是意外死亡盐固,警方通過查閱死者的電腦和手機屠橄,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評論 3 394
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來闰挡,“玉大人锐墙,你說我怎么就攤上這事〕ば铮” “怎么了溪北?”我有些...
    開封第一講書人閱讀 164,133評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我之拨,道長茉继,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,532評論 1 293
  • 正文 為了忘掉前任蚀乔,我火速辦了婚禮烁竭,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘吉挣。我一直安慰自己派撕,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,585評論 6 392
  • 文/花漫 我一把揭開白布睬魂。 她就那樣靜靜地躺著终吼,像睡著了一般。 火紅的嫁衣襯著肌膚如雪氯哮。 梳的紋絲不亂的頭發(fā)上际跪,一...
    開封第一講書人閱讀 51,462評論 1 302
  • 那天,我揣著相機與錄音喉钢,去河邊找鬼姆打。 笑死,一個胖子當(dāng)著我的面吹牛肠虽,可吹牛的內(nèi)容都是我干的穴肘。 我是一名探鬼主播,決...
    沈念sama閱讀 40,262評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼舔痕,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了豹缀?” 一聲冷哼從身側(cè)響起伯复,我...
    開封第一講書人閱讀 39,153評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎邢笙,沒想到半個月后啸如,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,587評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡氮惯,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,792評論 3 336
  • 正文 我和宋清朗相戀三年叮雳,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片妇汗。...
    茶點故事閱讀 39,919評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡帘不,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出杨箭,到底是詐尸還是另有隱情寞焙,我是刑警寧澤,帶...
    沈念sama閱讀 35,635評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站捣郊,受9級特大地震影響辽狈,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜呛牲,卻給世界環(huán)境...
    茶點故事閱讀 41,237評論 3 329
  • 文/蒙蒙 一刮萌、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧娘扩,春花似錦着茸、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至旋膳,卻和暖如春澎语,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背验懊。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評論 1 269
  • 我被黑心中介騙來泰國打工擅羞, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人义图。 一個月前我還...
    沈念sama閱讀 48,048評論 3 370
  • 正文 我出身青樓减俏,卻偏偏與公主長得像,于是被迫代替她去往敵國和親碱工。 傳聞我的和親對象是個殘疾皇子娃承,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,864評論 2 354

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