深入理解Android中的Window和WindowManager
Window表示一個(gè)窗口的概念,Window是一個(gè)抽象類,它的具體實(shí)現(xiàn)是PhoneWindow。創(chuàng)建一個(gè)Window,需要通過WindowManager即可完成,WindowManager是外界訪問Window的入口,Window具體實(shí)現(xiàn)位于WindowManagerService中,WindowManager和WindowManagerService的交互是一個(gè)IPC的過程。Android中,所有的視圖都是通過Window來呈現(xiàn),不管是Activity、Dialog、還是Toast,它們的視圖實(shí)際上都是附加在Window上,因此Window是實(shí)際View的直接管理者,單擊事件由Window傳遞給DecorView,然后再由DecorView傳遞給我們的View,就連Activity的設(shè)置視圖方法setContentView在底層也是通過Window來完成的。
Window和WindowManager
添加一個(gè)Window的過程,重點(diǎn)代碼是:
mWindowManager.addView(mFLoatingButton,mLayoutParams);
WindowManager.LayoutParams中有兩個(gè)flags和type參數(shù)。
Flags參數(shù)有三個(gè)Window屬性
- FLAG_NOT_FOCUSABLE。表示W(wǎng)indow不需要獲取焦點(diǎn),也不需要接收各種輸入事件,最終事件會(huì)直接傳遞給下層的具有焦點(diǎn)的Window
- FLAG_NOT_TOUCH_MODAL。在此模式下,系統(tǒng)會(huì)將當(dāng)前Window區(qū)域以外的單擊事件傳遞給底層的Window,當(dāng)前Window區(qū)域以內(nèi)的單擊事件則自己處理,這個(gè)標(biāo)記很重要,一般來說都需要開啟此標(biāo)記,否則其他Window將無法收到單擊事件
- FLAG_SHOW_WHEN_LOCKED。開啟此模式可以讓W(xué)indow顯示在鎖屏的界面上。
Type參數(shù)表示W(wǎng)indow的類型,有三種類型,分別是應(yīng)用Window,子Window和系統(tǒng)Window,應(yīng)用類Window對(duì)應(yīng)一個(gè)Activity,子Window不能單獨(dú)存在,它需要附屬在特定的父Window之中,比如常見的Dialog就是一個(gè)子Window,系統(tǒng)Window是需要聲明權(quán)限在能創(chuàng)建的Window,比如Toast和系統(tǒng)狀態(tài)欄這些都是系統(tǒng)Window。
Window是分層的,每個(gè)Window都有對(duì)應(yīng)的z-ordered,層級(jí)大的會(huì)覆蓋在層級(jí)小的Window的上面,在三類Window中,應(yīng)用類的Window的層級(jí)范圍是1-99,子Window的層級(jí)范圍是1000-1999,系統(tǒng)Window的層級(jí)的范圍是2000-2999,這些層級(jí)范圍對(duì)應(yīng)著WindowManager.LayoutParams的Type參數(shù)。如想要Window位于所有Window的最頂層,那么采用較大的層級(jí)即可。很顯然系統(tǒng)Window層級(jí)是最大的,而且系統(tǒng)層級(jí)有很多值。
WindowManager所提供的功能很簡(jiǎn)單,常用有三個(gè)方法,即添加View,更新View和刪除View,這三個(gè)方法定義在ViewManager中,而WindowManager繼承了ViewManager。
Window的內(nèi)部機(jī)制
Window是一個(gè)抽象的概念,每一個(gè)Window都對(duì)應(yīng)著一個(gè)View和一個(gè)ViewRootImpl,Window和View通過ViewRootImpl來建立聯(lián)系,說明View才是Window存在的實(shí)體,在實(shí)際使用中無法直接訪問Window,對(duì)Window的訪問必須通過WindowManager。
Window的添加過程
Window的添加過程需要通過WindowManager的addView來實(shí)現(xiàn),WindowManager是一個(gè)接口,它的真正實(shí)現(xiàn)是WindowManagerImpl類。
@Override public void addView(View view,ViewGroup.LayoutParams params){ mGlobal.addView(view,params,mDisplay,mParentWindow); } @Override public void updateViewLayout(View view,ViewGroup.LayoutParams params){ mGlobal.updateViewLayout(view,params); } @Override public void removeView(View view){ mGlobal.removeView(view,false); }
可以看到,WindowManagerImpl并沒有直接實(shí)現(xiàn)Window的三大操作,而是全部交給了WindowManagerGlobal來處理,WindowManagerGlobal以工廠的形式向外提供自己的實(shí)例。WindowManagerGlobal的addView方法主要分為如下幾步:
- 檢查參數(shù)是否合法,如果是子Window那么還需要調(diào)整一些布局參數(shù)
- 創(chuàng)建ViewRootImpl并將View添加到列表中
- 通過ViewRootImpl來更新界面并完成Window的添加過程
Window的刪除過程
Window的刪除過程和添加過程一樣,都是先通過WindowManagerImpl后,在進(jìn)一步通WindowManagerGlobal來實(shí)現(xiàn)的。里面用到一個(gè)dispatchDetachedFromWindow方法內(nèi)部實(shí)現(xiàn),這個(gè)方法主要做了四件事:
垃圾回收相關(guān)的工作,比如清除數(shù)據(jù)和消息、移除回調(diào)
通過Session的remove方法刪除Window
調(diào)用View的dispatchDetachedFromWindow方法,在內(nèi)部會(huì)調(diào)用View的onDetachedFromWindow()以及onDetachedFromWindowInternal()
調(diào)用WindowManagerGlobal的doRemoveView方法刷新數(shù)據(jù)
Window的更新過程
主要是用到updateViewLayout方法,首先它需要更新View的LayoutParams并替換掉老的LayoutParams,接著再更新ViewRootImpl中的LayoutParams,這一步是通過ViewRootImpl的setLayoutParams方法來實(shí)現(xiàn)的。在ViewRootImpl中會(huì)通過scheduleTraversals方法對(duì)View進(jìn)行重新布局,包括測(cè)量、布局、重繪這三個(gè)過程。
Window的創(chuàng)建過程
View是Android中的視圖呈現(xiàn)方式,但是View不能單獨(dú)存在,它必須附著在Window這個(gè)抽象的概念上面,因此有視圖的地方就有Window。
Activity的Window創(chuàng)建過程
如何創(chuàng)建,需要了解Activity啟動(dòng)過程,比較復(fù)雜,但它最終由ActivityThred中的perfromLaunchActivity()來完成整個(gè)啟動(dòng)過程,在這個(gè)方法內(nèi)部會(huì)通過類加載器創(chuàng)建Activity的實(shí)例對(duì)象,并調(diào)用其attach方法為其關(guān)聯(lián)運(yùn)行過程中所依賴的一系列上下文環(huán)境變量。
在Activity的attach方法里,系統(tǒng)會(huì)創(chuàng)建Activity所屬的Window對(duì)象并為其設(shè)置回調(diào)接口,Window對(duì)象的創(chuàng)建是通過PolicyManager的makeNewWindow方法實(shí)現(xiàn)的,對(duì)于Activity的setContentView的實(shí)現(xiàn)可以看出,Activity將具體實(shí)現(xiàn)交給了Window處理,而Window的具體實(shí)現(xiàn)是PhoneWindow,所以只需要看PhoneWindow相關(guān)邏輯即可,大致以下幾個(gè)步驟:
- 如果沒有DecorView,那么就創(chuàng)建它。DecorView是一個(gè)FrameLayout,是Activity的頂級(jí)View,一般來說它的內(nèi)部包含標(biāo)題欄和內(nèi)部欄。
- 將View添加到DecorView的mContentParent中。
- 回調(diào)Activity的onContentChanged方法通知Activity視圖已經(jīng)發(fā)生改變。Activity的onContentChanged是一個(gè)空實(shí)現(xiàn)。
經(jīng)過上面三個(gè)步驟,DecorView已經(jīng)被創(chuàng)建初始化完畢,Activity的布局文件已經(jīng)成功添加到了DecorView的mContentParent中,但是這個(gè)時(shí)候DecorView還沒有被WindowManager正式添加到Window中,真正被視圖調(diào)用是在Activity的onResume方法,接著會(huì)調(diào)用Activity的makeVisible(),正是在makeVisible方法中,DecorView真正地完成了添加和顯示這兩個(gè)過程。
Dialog的Window創(chuàng)建過程
Dialog的Window的創(chuàng)建過程和Activity類似,有以下幾個(gè)步驟:
- 創(chuàng)建Window。同樣是通過PolicyManager的makeNewWindow方法來完成的。
- 初始化DecorView并將Dialog的視圖添加到DecorView中。
- 將DecorView添加到Window中并顯示。在Dialog的show方法中,會(huì)通過WindowManager將DecorView添加到Window中。
普通的Dialog有一個(gè)特殊之處,那就是必須采用Activity的Context,如果采用Application的Context,就會(huì)報(bào)錯(cuò)。
Toast的Window創(chuàng)建過程
Toast和Dialog不同,它的工作過程稍微復(fù)雜。首先Toast也是基于Window來實(shí)現(xiàn)的,但是由于Toast具有定時(shí)取消這一功能,所以系統(tǒng)采用了Handler。在Toast的內(nèi)部有兩類的IPC過程,第一類是Toast訪問NotificationManagerService,第二類是NotificationManagerService回調(diào)Toast的TN接口。
Toast屬于系統(tǒng)Window,它內(nèi)部的視圖有兩種方式指定,一種是系統(tǒng)默認(rèn)的樣式,另一種是通過setView方法來指定一個(gè)自定義View,不管如何,它們都對(duì)應(yīng)Toast的一個(gè)View類型的內(nèi)部成員mNextView。Toast提供了show和cancel分別用于顯示和隱藏Toast,它們的內(nèi)部是一個(gè)IPC過程。
以上所述是小編給大家介紹的Android中的Window和WindowManager,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
Android RecyclerView item選中放大被遮擋問題詳解
這篇文章主要介紹了Android RecyclerView item選中放大被遮擋問題詳解,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-04-04Android解決getExternalStorageDirectory在29后廢棄問題(推薦)
這篇文章主要介紹了Android解決getExternalStorageDirectory在29后廢棄問題(推薦),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-02-02快速解決設(shè)置Android 23.0以上版本對(duì)SD卡的讀寫權(quán)限無效的問題
今天小編就為大家分享一篇快速解決設(shè)置Android 23.0以上版本對(duì)SD卡的讀寫權(quán)限無效的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-08-08Android+Flutter實(shí)現(xiàn)彩虹圖案的繪制
彩虹,是氣象中的一種光學(xué)現(xiàn)象,當(dāng)太陽光照射到半空中的水滴,光線被折射及反射,在天空上形成拱形的七彩光譜。接下來,我們就自己手動(dòng)繪制一下彩虹圖案吧2022-11-11Kotlin學(xué)習(xí)教程之協(xié)程Coroutine
這篇文章主要給大家介紹了關(guān)于Kotlin學(xué)習(xí)教程之協(xié)程Coroutine的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-05-05FragmentTabHost FrameLayout實(shí)現(xiàn)底部導(dǎo)航欄
這篇文章主要為大家詳細(xì)介紹了FragmentTabHost和FrameLayout實(shí)現(xiàn)底部導(dǎo)航欄,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-03-03Android編程使用自定義View實(shí)現(xiàn)水波進(jìn)度效果示例
這篇文章主要介紹了Android編程使用自定義View實(shí)現(xiàn)水波進(jìn)度效果,結(jié)合實(shí)例形式詳細(xì)分析了Android水波動(dòng)畫效果的具體實(shí)現(xiàn)步驟與相關(guān)注意事項(xiàng),需要的朋友可以參考下2017-01-01解決Android自定義view獲取attr中自定義顏色的問題
這篇文章主要介紹了Android自定義view獲取attr中自定義顏色的問題解決方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-12-12Android實(shí)現(xiàn)類似網(wǎng)易新聞選項(xiàng)卡動(dòng)態(tài)滑動(dòng)效果
這篇文章主要介紹了Android實(shí)現(xiàn)類似網(wǎng)易新聞選項(xiàng)卡動(dòng)態(tài)滑動(dòng)效果的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-11-11