View.post() 不靠譜的地方你知道多少
一、前言
有時(shí)候,我們會(huì)需要用到 View.post() 方法,來將一個(gè) Runnable 發(fā)送到主線程去執(zhí)行。這一切,看似很美好,它最終會(huì)通過一個(gè) Handler.post() 方法去執(zhí)行,又避免我們重新定義一個(gè) Handler 對象。
但是,從 Android 7.0(Api level 24) 開始,View.post() 將不再那么靠譜了,你 post() 出去的 Runnable ,可能永遠(yuǎn)也不會(huì)有機(jī)會(huì)執(zhí)行到。
二、post 在 7.0 的差異
2.1 post 方法的差異
前面提到,這個(gè)問題只出現(xiàn)在 Android 7.0 上。那么就先從源碼分析 Android 7.0 到底對 View.post() 做了什么改動(dòng)。

用 Diff 看一下它們的差異,左邊是 Api Level 24+(以下簡稱 Api24) 的代碼,右邊是 Api level 23-(以下簡稱 Api23) 的代碼。
很明顯的可以看出來,它們只有在 mAttachInfo 為 null 的時(shí)候,執(zhí)行的邏輯才會(huì)有差異。
Api24 中,會(huì)調(diào)用 getRunQueue().post(action),而 Api23 會(huì)調(diào)用 ViewRootImpl.getRunQueue().post(action) 方法,他們的差異就在這里。
2.2 Api23 post 的細(xì)節(jié)
先簡單理解一下,ViewRootImpl 是什么。
ViewRootImpl 可以理解是一個(gè) Activity 的 ViewTree 的根節(jié)點(diǎn)的實(shí)例。每個(gè) ViewRootImpl 就是用來管理 DecorView 和 ViewTree。
ViewRootImpl 中的用來承載 Runnable 的隊(duì)列是 sRunQueues ,它一個(gè)靜態(tài)的變量,也就是說在 App 的生命周期內(nèi),ViewRootImpl 中的這個(gè)消息隊(duì)列都是同一個(gè)。
再來看看前面提到的 ViewRootImpl.getRunQueue().post() 到底干了什么?

post() 方法只是單純的將它包裝成一個(gè) HandlerAction 對象,然后放入 mActions 這個(gè) ArrayList 中。繼續(xù)追查下去就需要知道 mActions 中添加的 HandlerAction 在何時(shí)被消費(fèi)掉了。
消費(fèi) HandlerAction 的地方,是 executeActions() 方法。

它最終,還是調(diào)用的 handler.postDelayed() ,這沒什么好說的,關(guān)鍵點(diǎn)在于 executeAction() 方法,是在什么時(shí)候被調(diào)用的。
executeAction() 是被 TraversalRunnable 調(diào)用 doTraversa() ,在doTraversa() 方法中,進(jìn)行調(diào)用的。而 TraversalRunnable 又是通過 Choreographer.postCallBack() 去循環(huán)調(diào)用的。這個(gè) Choreographer 通過 doScheduleCallback() 發(fā)送一個(gè) MSG_DO_SCHEDULE_CALLBACK 類型的消息循環(huán)調(diào)用,間隔就是一個(gè) VSync 的間隔。
關(guān)于 Choreographer ,不是本文的重點(diǎn),有興趣可以單獨(dú)了解一下。
所以,在 Api23 以下,executeAction() 是會(huì)被循環(huán)調(diào)用,基本上其內(nèi)的 mActions 只要有未執(zhí)行的 Runnable 立刻就會(huì)被消費(fèi)掉。
所以在 Api23 以下的設(shè)備上,View.post() 基本上是靠譜的,post 出去的 Runnable 都會(huì)有機(jī)會(huì)執(zhí)行到。
2.3 Api24 的細(xì)節(jié)
再來看看在 Api24 中的實(shí)現(xiàn)細(xì)節(jié),在 Api24 中,調(diào)用的是 getRunQueue().post() 方法,它操作的是一個(gè) HandlerActionQueue 對象。

內(nèi)部的結(jié)構(gòu)其實(shí)和 Api23 很像,也是維護(hù)了一個(gè) HandlerAction 的數(shù)組 mActions 。
最終消費(fèi)掉 mActions 的地方,依然是一個(gè) executeActions() 方法。

回到根本的問題,executeActions() 方法在什么時(shí)機(jī)會(huì)被調(diào)用到,繼續(xù)追查可以看到它在 View.dispatchAttachedToWindow() 方法中,會(huì)被調(diào)用。

既然,executeActions() 方法,在 Api24 及以上,只會(huì)在 dispatchAttachedToWindow() 的方法中,才有機(jī)會(huì)被調(diào)用到,而 View.dispatchAttachedToWindow() 方法,只有在這個(gè) View 通過 addView() 等方法,加入到一個(gè) ViewGroup 的時(shí)候,才會(huì)被調(diào)用到。這就導(dǎo)致寫在 Layout 布局中的控件,是不會(huì)有機(jī)會(huì)再調(diào)用 addView() 方法的,所以它永遠(yuǎn)也得不到執(zhí)行。這也就到時(shí)了 Api24 下,View.post() 表現(xiàn)的現(xiàn)象不一致的緣故。
三、小結(jié)
View.post() 方法,在不同版本的差異,根本原因還是在于 Api23 和 Api24 中,executeActions() 方法的調(diào)用時(shí)機(jī)不同,導(dǎo)致 View 在沒有 mAttachInfo 對象的時(shí)候,表現(xiàn)不一樣了。
所以我們在使用的過程中需要慎用,區(qū)分出實(shí)際使用的場景,一般規(guī)范自己的代碼即可:
在 View 已經(jīng)被顯示出來之后,再調(diào)用 View.post() 方法(這個(gè)時(shí)候 mAttachInfo 已經(jīng)不為空了)。
盡量避免使用 View.post() 方法,可以直接使用 Handler.post() 方法來替代。
總結(jié)
以上所述是小編給大家介紹的View.post() 不靠譜的地方,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
JS實(shí)現(xiàn)圖片高亮展示效果實(shí)例
這篇文章主要介紹了JS實(shí)現(xiàn)圖片高亮展示效果的方法,實(shí)例分析了JavaScript響應(yīng)鼠標(biāo)事件動(dòng)態(tài)操作頁面元素樣式的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-11-11
JavaScript實(shí)現(xiàn)點(diǎn)擊按鈕復(fù)制指定區(qū)域文本(推薦)
這篇文章主要介紹了JavaScript實(shí)現(xiàn)點(diǎn)擊按鈕復(fù)制指定區(qū)域文本(推薦)的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-11-11
js日期相關(guān)函數(shù)dateAdd,dateDiff,dateFormat等介紹
這篇文章主要介紹了js日期相關(guān)函數(shù)dateAdd,dateDiff,dateFormat等介紹,需要的朋友可以參考下2016-09-09
JS實(shí)現(xiàn)判斷兩個(gè)日期不能跨年和跨月
這篇文章主要為大家詳細(xì)介紹了如何利用JavaScript語言實(shí)現(xiàn)判斷兩個(gè)日期不能跨年和跨月,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2023-04-04
php對mongodb的擴(kuò)展(初識(shí)如故)
mongodb的數(shù)據(jù)存儲(chǔ)格式是一種由于MongoDB的文檔結(jié)構(gòu)為BJSON格式(BJSON全稱:Binary JSON),而BJSON格式本身就支持保存二進(jìn)制格式的數(shù)據(jù),因此可以把文件的二進(jìn)制格式的數(shù)據(jù)直接保存到MongoDB的文檔結(jié)構(gòu)中2012-11-11
text-align:justify實(shí)現(xiàn)文本兩端對齊 兼容IE
對于text-align 我們再熟悉不過了,可是它有個(gè)justify屬性,平時(shí)很少用到,就鮮為人知了。justify是一種文本靠兩邊布局方式,一般應(yīng)用于書刊雜志排版;合理運(yùn)用text-align:justify 有時(shí)會(huì)省去很多開發(fā)的時(shí)間,通過本文介紹text-align:justify實(shí)現(xiàn)文本兩端對齊 兼容IE2015-08-08

