胡琪

为今天工作,为明天投资,为未来孵化一些东西!

简单谈谈安卓中的窗口管理系统

安卓中的窗口管理系统是一个非常复杂的系统,涉及到的对象概念非常多,主要包括Activity,Window,DecorView,WindowManager,WindowManagerService等,这里不对窗口管理系统整体做分析,主要来看看Activity,Window,DecorView以及开发者在Activity中通过setContentView设置的xml文件中配置的ContentView(估且这么叫)之间的层级关系,因为这一块的内容既与开发相关又不太过于底层,但是又能帮助我们从宏观上了解安卓的窗口系统。

  1. Window是Activity的一个成员,Window本身只是一个抽象类,具体的实现是PhoneWindow
  2. DecorView是PhoneWindow的一个内部类,本质是一个FrameLayout的ViewGroup
  3. ContentView是DecorView内部的一个子View,DecorView的子View除了ContentView外还包括与具体主题相关的标题栏等子View
  4. 开发者在xml中设置的界面会被添加到ContentView中作为子View

下面通过代码的方式来论证上述结论。

首先看看Activity,在Activity中有如下的代码片段:

从上述代码片段可以看到每一个Activity内部都持有一个mWindow的Window对象,然后我们看看Window的部分代码片段

从这里可以看到Window仅仅只是一个抽象类而已,而且Activity中一些重要的方法实质上是通过调用Window类来实现的,如setContentView等,具体的实现是PhoneWindow类,我们看下PhoneWindow部分代码片段:

可以看到PhoneWindow内部持有一个DecorView对象,我们来看下DecorView的定义:

从这里可以看到DecorView本质是一个FrameLayout,即DecorView是一个ViewGroup容器。注意PhoneWindow不是View,PhoneWindow中关于View的操作很多都是通过DecorView来完成添加的。到这里Activity,Window,DecorView之间的层级关系就非常清楚了,但是开发者通过setContentView在xml文件中定义的界面与Activity,Window,DecorView之间关系是怎样的呢?这个就需要我们从setContentView的源码来分析了。首先看下Activity的setContentView函数

从这里可以看到Activity的setContentView最终是通过PhoneWindow来完成的,因为getWindow()返回的是一个Window对象。所以我们接下来看下PhoneWindow的setContentView函数。

注意这里的 installDecor()函数,这个函数非常重要,该函数做了两件非常重要的事,一个是DecorView的创建,另一个就是将android.R.id.content 的布局(本质为FrameLayout)与mContentParent关联起来。关于installDecor()函数的源码分析,大家可以参考这个,写的很不错:http://blog.csdn.net/yanbober/article/details/45970721/ 。注意另外一个很重要的语句 mLayoutInflater.inflate(layoutResID, mContentParent);该语句将id为layoutResID的布局文件,即开发者在xml中配置的文件,转化为View树然后添加到mContentParent中。

到这里Activity,Window,DecorView,ContentView,xml文件之间的层级关系就非常清楚了。即Window为Activity的成员,DecorView是PhoneWindow的内部类成员,ContentView是DecorView的子View,xml文件转化的View是ContentView的子View。了解它们之间的层级关系之后要获取xml文件中的根布局就很容易了。代码如下:

如果要获取根布局下的全部子View可以使用如下代码:

 

关于安卓窗口系统的详细讲解大家可以参看以下文章:

http://blog.csdn.net/yanbober/article/details/45970721/

http://www.jianshu.com/p/851714f5e640

https://zhuanlan.zhihu.com/p/26834562

打赏

点赞