Android開發藝術探索
第八章理解Window和WindowManager+第九章四大組件的工作過程
- 什么是Window?
答:Android中的所有視圖都是通過Window呈現的,不管是Activity、Dialog還是Toast,他們的視圖都是附加在Window上的,因此Window是視圖的直接管理者 - Window、WindowManager和WindowManagerService三者之間的關系?
答:Window是一個抽象類,PhoneWindow是他的實現類,通過WindowManager可以創建一個Window。WindowManager的具體實現在WindowManagerService中,WindowManagerService和WindowManager的交互過程是IPC過程 - Window的類型有幾種?
答:三種
應用Window:一個Activity
子Window:Dialog
系統Window:Toast和狀態欄,需要申明權限才能創建 - Window是分層的,每個Window都有對應的層級,大層級可以覆蓋在小層級Window上面
- WindowManager中的方法很簡單,添加View,更新View和刪除View
- Window的組成是什么?
答:每個Window對應一個View和一個ViewRootImpl,Window和View通過ViewRootImpl建立聯系 - Window不是實際存在的,它是以VIew的形式存在的。View才是Window存在的實體。
- 簡述Window的添加過程
答:Window的添加過程是通過WindowManager實現的,WindowManager是一個抽象類,實現類是WindowManagerImpl。在實現類中,addView方法是通過WindowManagerGlobal處理的,主要分為一下幾步:
①檢查參數是否合法,如果子Window還要進行進一步調整(參數非空性檢測,子Window調整大小,如Dialog)
②創建ViewRootImpl并將View添加到鏈表中
③通過VIewRootImpl來更新界面并完成Window的繪制過程
View的繪制過程就是通過ViewRootImpl來完成的
④接著調用WindowSession完成Window的添加過程(繪制在應用進程,添加在系統進程),Session內部會通過WindowManagerService實現Window的添加(IPC方式) - 簡述Window的刪除過程
答:前面和添加過程差不多,WindowManagerImpl-->WindowManagerGlobal
①通過findViewLocker查找待刪除View的索引
②調用remove方法刪除
主要通過die方法完成,die方法只是發送一個請求刪除的消息后就立刻返回了,如果是異步刪除,發送MSG_DIE的消息,VIewRootImpl中的Handler會處理此消息并調用doDie方法,同步消息直接調用doDie方法。doDie方法中主要做四件事:
(1)垃圾回收工作,清除數據,消息等
(2)通過Session的remove方法刪除Window
(3)調用View的dispatchDetachedFromWindow方法(View從Window移除時就會調用),做一些資源回收工作,中止動畫、停止線程等
(4)調用WindowManagerGlobal的doRemove方法刷新數據 - 簡述Window的更新過程
答:更新VIew+Window
更新View的布局參數,這一步是通過VIewRootImpl的setLayoutParams方法實現的,測量、布局、重繪三個過程;更新Window視圖是通過Session實現的 - View是Android中視圖的呈現方式,但是View不能單獨存在,他必須依靠Window上面
- Window的創建過程
答:Activity創建時,首先通過類加載器創建Activity的實例對象,然后調用attach方法為其關聯運行過程中所依賴的一系列上下文環境,在attach方法里:
系統會創建Activity所屬的Window對象并為其設置回調接口(onAttachedToWindow,onDetachedFromWindow、dispatchToucheEvent等) - setContentView(Activity創建)的過程
答:setContenView是通過PhoneWindow實現的,具體實現過程如下:
(1)如果沒有DecorView就創建它
(2)將View添加到DecorView的mContentParent中
(3)回調onContentChanged方法通知Activity視圖發生改變
注意:這個時候DecorVIew并沒有被WindowManager正式添加到Window中,因為DecorView還沒有被Window識別(也就是沒顯示到前臺),在onResume方法中才真正完成了添加和顯示,才能被用戶看到 - Dialog的創建過程
答:和Activity的創建過程類似,也是先創建DecorView,添加View并顯示 - Toast的Window的創建過程
答:不同之處:
Toast的內部有兩類IPC過程:①Toast訪問NotificationManagerService ②NotificationManagerService回調Toast里的TN接口。TN類中有兩個方法show和hide,因為是NMS跨進程調用,他們運行在Binder線程池中,所以在執行過程中需要切換線程(內部用Handler)實現。所以Toast無法在沒有Looper的線程中彈出
問題:
- view異步刪除和同步刪除的區別
答:異步刪除,發送msg消息,VIewRootImpl中的Handler來處理;同步刪除,直接刪除 - View的dispatchDetachedFromWindow方法有什么用?
答:在內部會調用onDetachedFromWindow方法,當View從Window移除時,這個方法就會調用,可以在這個方法內做一些資源回收工作,比如終止動畫,停止線程等 - onAttachedToWindow,onDetachedFromWindow、dispatchToucheEvent各自的作用是什么
答:onAttachedToWindow:當VIew添加add到Window時回調
onDetachedFromWindow:當View從Window移除時回調
dispatchTouchEvent:對觸摸事件進行分發,返回值根據onTouchEvent(自身攔截)或child.dispatchTouchEvent決定 - Toast和Dialog的區別
答:Dialog是子Window,必須依賴父Window
Toast是系統Window - Activity的Window創建過程有沒有IPC通信?有 在attach中,但只有添加View的操作中沒有