Window和Window内部机制
Window表示一个窗口的概念。Android所有的View都是是通过Window呈现的,比如Activity,Dialog,Toast等。如果有在桌面上创建一个悬浮窗的需求,就需要用到Window.Window也是View的直接管理者。
Window是一个抽象类,PhoneWindow是其具体实现。Windwo创建通过WindowManager,而具体实现是在WindowManagerService中,这两者交互是个IPC过程。
WindowManager常用的功能:
addView(),updateViewLayout(),removeView();
Window内部机制
Window是一个抽象的概念,每一个Window都对应着一个View和ViewRootImpl。Window和View通过ViewRootImpl建立联系。所以Window不是单独存在的,他是以View的形式存在。
Window的添加过程
Window添加过程,通过WindowManager的addView,而WindowManager是一个接口,真正实现它的是WindowManagerImpl。在WindowManagerImpl的三大操作中(增,更,查),全部委托给WindowManagerGlobal来实现。
在WindowManagerGlobal中创建ViewRootImpl并将View添加到列表中。
通过ViewRootImpl来更细页面,在通过WindowSession(IWindowSession),它是一个Binder对象。这就是一次IPC调用。在Session内部通过WindowManagerService来实现Window的添加。
Window的删除过程
通过WindowMImpl,在通过WindowMGlobal中的removeView实现,在这remove是通过ViewRootImpl来完成删除操作的。
Window的更新过程
通过WindowMImpl,在通过WindowMGlobal中的updateViewLatyout实现,在通过更新ViewRootImpl中的LayoutParams,通过setLayoutParams方法来实现。这个过程最终是由WindowManagerService的relayoutWindow()来具体实现,也是一个IPC过程。
Window的创建过程
Activity的Window创建过程
在Activity的attach方法房中,系统会创建Activity所属的Window对象并为其设置回调接口。Window对象是通过PolicyManager的makeNewWindow方法实现的。PolicyManager真正实现的是Policy类。
Policy类中的makeNewWindow方法中可知Window的具体实现是PhoneWindow。setContentView实际是由PhoneWindow的setContentView来实现
PhoneWindow中setContentView的步骤:
1.没有DecorView就创建一个。
2.将View添加到DecorView的mContentParene中
3.回调Activity的onContentChanged方法通知Activity视图已经发生改变
Dialog的Window创建过程
1.创建Window和Activity的Window创建过程类似,也是通过PolicyManager的makeNewWindow方法实现的。
2.初始化DecorView将Dialog的视图添加到DecorView中
3.通过WindowManager将DecorView添加到Window中并显示,
Toast的Window创建过程
Toast的显示和隐藏都是通过NotificationManagerService来实现,NMS运行在系统进程中。Toast和NMS进行IPC过程时,当NMS处理Toast的显示或者隐藏会跨进程回调TN(一个Binder类)中的方法。
Toast的显示过程:首先调用NMS中enqueueToast,enqueueToast首先将Toast请求封装为ToastRecord对象并添加到mToastQueue队列中,然后NMS就会通过showNextToastLocked方法来显示当前的Toast,Toast显示是由ToastRecord的callback来完成的。这个callback实际就是Toast中的TN对象的远程Binder。
Toast的隐藏过程也是通过ToastRecord的callback完成,同样也是一个IPC过程。
在TN的handleShow中会将Toast的视图添加到Window中,在handleHide中会将Toast视图从Window中移除。
一个应用中的Activity的个数:
Window有三种类型:应用Window、子Window和系统Window。应用Window对应着一个Activity,子Window不能单独存在,需要附属在特定的Window中,比如Dialog。系统Window需要声明权限才能创建Window,比如Toast和系统状态栏。
所以应用中Window的个数是:Activity数量+Dialog数量+使用的Toast数量+悬浮窗个数。