Android消息循環(huán)機制(一)

Android的Handler是什么?網(wǎng)上很多資料都會介紹說Android Handler是用來處理跨線程通信城瞎,跨線程更新UI的渤闷,這么說肯定沒錯,但這只是Handler的作用脖镀,個人覺得并沒有說到核心點上飒箭。

跨線程通信,Java的內(nèi)存模型本身就已經(jīng)實現(xiàn)了蜒灰,在java里跨線程通信非常簡單弦蹂,直接引用變量就可以了,當(dāng)然如果需要處理線程安全問題用synchronized或者volatile處理就可以了强窖。 為什么還要用Handler凸椿?單純說Handler跨線程通信,其實是不合適的翅溺。
那Handler到底是什么削饵?Handler其實是Android的消息循環(huán)模型。
看一個最普通的java代碼未巫。

public static void main(String[] args){
    System.out.println("Hello World");
}

一共三行窿撬。JVM虛擬機可能是這樣處理的。

-->>創(chuàng)建線程
-->>執(zhí)行1叙凡,2劈伴,3行
-->>代碼結(jié)束,線程結(jié)束

那問題來了握爷,我們App如果也是類似上面的代碼的話就會有問題了跛璧,一個最簡單的HelloWorld App并不會在繪制完HelloWrold TextView之后就結(jié)束線程。那怎么辦新啼?
AndroidSdk里有個類ActivityManagerService追城,里面有個startProcessLocked方法,有類似下面的代碼燥撞。

if (entryPoint == null) entryPoint = "android.app.ActivityThread";
Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "Start proc: " +
        app.processName);
checkTime(startTime, "startProcess: asking zygote to start proc");
Process.ProcessStartResult startResult = Process.start(entryPoint,
        app.processName, uid, uid, gids, debugFlags, mountExternal,
        app.info.targetSdkVersion, app.info.seinfo, requiredAbi, instructionSet,
        app.info.dataDir, entryPointArgs);

第一行的"android.app.ActivityThread"其實就是App的入口類座柱,在ActivityThread,里面有個main方法物舒。這個方法就是整個App的入口色洞。

public static void main(String[] args) {
    ......
    Looper.prepareMainLooper();
    ......
    Looper.loop();
    throw new RuntimeException("Main thread loop unexpectedly exited");
}

這段代碼最后一行很有意思,是一個異常冠胯,說明如果代碼跑到這里App就崩潰了火诸。我們App既然沒有崩潰,那就表示這個代碼從來沒有被執(zhí)行到荠察。怎么才能不被執(zhí)行到置蜀?死循環(huán)奈搜。死循環(huán)只要不發(fā)生異常是永遠(yuǎn)不會結(jié)束的。跟進去發(fā)現(xiàn)確實是一個死循環(huán)盯荤。

public static void loop() {
    ......
    for(;;){
      ......
    }
    ......
}

死循環(huán)如果不停的去輪詢一個變量媚污,檢查狀態(tài)獲取消息,然后需要處理數(shù)據(jù)了廷雅,就修改被輪詢的對象應(yīng)該也是可以的耗美,但有個很嚴(yán)重的問題,這樣不加節(jié)制的死循環(huán)航缀,即使當(dāng)前沒有任何消息需要處理也會吃滿CPU商架,那為了不必要的CPU消耗,我們肯定是希望有個東西能通知我們什么時候有消息了芥玉,什么時候再喚醒才最好蛇摸。我們先看下Android里,App在安靜狀態(tài)代碼都在干嘛灿巧。新建一個HelloWorld項目赶袄,跑起來,然后進度debug模式抠藕,在dubug模式里饿肺,點擊下圖里的紅色框中的按鈕,這個按鈕可以dump當(dāng)前App的堆棧數(shù)據(jù)盾似。


dump_stack_0.png

打印主線程堆棧狀態(tài):


dump_stack_1.png

無論dump多少次敬辣,只要App是安靜狀態(tài),沒有處理任何事件就永遠(yuǎn)是上面的代碼零院。上面的結(jié)果是什么意思溉跃?上面的結(jié)果表示App在安靜的時候是阻塞在了
android.os.MessageQueue.nativePollOnce(Native Method)
android.os.MessageQueue.next(MessageQueue.java:323)

我們先查看下App的CPU占用,只要是安靜狀態(tài)CPU占用就一直為0告抄。


安靜狀態(tài)CPU占用.png

這表示Android并沒有空轉(zhuǎn)CPU撰茎,用腳趾頭想其實也不可能空轉(zhuǎn),一直空轉(zhuǎn)那CPU還不得罵人了打洼。跟進MessageQueue的next方法龄糊,for死循環(huán)調(diào)用的nativePollOnce(ptr, nextPollTimeoutMillis);是個Native方法,是用C實現(xiàn)的拟蜻,背后的原理其實是利用的Linux的epoll機制绎签。有興趣的可以搜epoll的相關(guān)資料枯饿。

Android整個消息過程大致可以理解為這樣:
A酝锅、主線程
-->>Android App進程創(chuàng)建
-->>初始化創(chuàng)建一個消息隊列
-->>主線程死循環(huán)讀取消息隊列
-->>消息處理完nativePollOnce掛起等待新消息

B、其他線程
-->>直接引用主線程的消息隊列奢方,添加消息到隊列里搔扁。(由于做了線程同步這時候主線程的隊列數(shù)據(jù)已經(jīng)和子線程一致了)
-->>調(diào)用nativeWake通知Native喚醒主線程

C爸舒、主線程
-->>主線程nativePollOnce掛起取消
-->>處理消息
-->>消息處理完消息,nativePollOnce繼續(xù)掛起等待新消息
......
無限循環(huán)
……

如果希望一個Runnable在主線程執(zhí)行稿蹲,那很簡單:
在任意線程(可以是主線程也可以不是主線程)創(chuàng)建Runnable扭勉,然后在任意線程將Runnable添加到主線程的消息隊列,然后調(diào)用nativeWake通知Native喚醒主線程苛聘,主線程喚醒涂炎,然后主線程調(diào)用Runnable。注意這時候Runnable就已經(jīng)是主線程在調(diào)用執(zhí)行了设哗。
這就是Handler的大致工作過程唱捣,Android就是通過Handler來實現(xiàn)消息循環(huán)機制的。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末网梢,一起剝皮案震驚了整個濱河市震缭,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌战虏,老刑警劉巖拣宰,帶你破解...
    沈念sama閱讀 223,002評論 6 519
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異烦感,居然都是意外死亡巡社,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,357評論 3 400
  • 文/潘曉璐 我一進店門手趣,熙熙樓的掌柜王于貴愁眉苦臉地迎上來重贺,“玉大人,你說我怎么就攤上這事回懦∑希” “怎么了?”我有些...
    開封第一講書人閱讀 169,787評論 0 365
  • 文/不壞的土叔 我叫張陵怯晕,是天一觀的道長潜圃。 經(jīng)常有香客問我,道長舟茶,這世上最難降的妖魔是什么谭期? 我笑而不...
    開封第一講書人閱讀 60,237評論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮吧凉,結(jié)果婚禮上隧出,老公的妹妹穿的比我還像新娘。我一直安慰自己阀捅,他們只是感情好胀瞪,可當(dāng)我...
    茶點故事閱讀 69,237評論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般凄诞。 火紅的嫁衣襯著肌膚如雪圆雁。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,821評論 1 314
  • 那天帆谍,我揣著相機與錄音伪朽,去河邊找鬼。 笑死汛蝙,一個胖子當(dāng)著我的面吹牛烈涮,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播窖剑,決...
    沈念sama閱讀 41,236評論 3 424
  • 文/蒼蘭香墨 我猛地睜開眼跃脊,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了苛吱?” 一聲冷哼從身側(cè)響起酪术,我...
    開封第一講書人閱讀 40,196評論 0 277
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎翠储,沒想到半個月后绘雁,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,716評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡援所,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,794評論 3 343
  • 正文 我和宋清朗相戀三年庐舟,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片住拭。...
    茶點故事閱讀 40,928評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡挪略,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出滔岳,到底是詐尸還是另有隱情杠娱,我是刑警寧澤,帶...
    沈念sama閱讀 36,583評論 5 351
  • 正文 年R本政府宣布谱煤,位于F島的核電站摊求,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏刘离。R本人自食惡果不足惜室叉,卻給世界環(huán)境...
    茶點故事閱讀 42,264評論 3 336
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望硫惕。 院中可真熱鬧茧痕,春花似錦、人聲如沸恼除。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,755評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至埃脏,卻和暖如春搪锣,著一層夾襖步出監(jiān)牢的瞬間秋忙,已是汗流浹背彩掐。 一陣腳步聲響...
    開封第一講書人閱讀 33,869評論 1 274
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留灰追,地道東北人堵幽。 一個月前我還...
    沈念sama閱讀 49,378評論 3 379
  • 正文 我出身青樓,卻偏偏與公主長得像弹澎,于是被迫代替她去往敵國和親朴下。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,937評論 2 361

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