在Activity
的attach()
方法里,系统会创建Activity
所属的Window
对象并为其设置回调接口,Window
对象的创建是通过PolicyManager
的makeNewWindow
方法实现的。由于Activity
实现了Window
的Callback
接口,因此当Window
接收到外界的状态改变时就会回调Activity
的方法。Callback
接口中的方法很多,但是有几个却是我们都非常熟悉的,比如onAttachedToWindow
,onDetachedFromWindow
,diapatchTouchEvent
,等等
// 创建 Window 对象
mWindow = PolicyManager.makeNewWindow(this);
// 实现 Window 的 Callback 接口
mWindow.setCallback(this);
mWindow.setOnWindowDismissedCallback(this);
. . .
从上面的分析可以看出,Activity
的Window
是通过PolicyManager
的一个工厂方法来创建的,但是从PolicyManager
的类名可以看出,它不是一个普通的类,它是一个策略类。PolicyManager
中实现的几个工厂方法全部在策略接口IPolicy
中声明了。
public interface IPolicy {
public Window makeNewWindow(Context context);
public LayoutInflater makeNewLayoutInflater(Context context);
public WindowManagerPolicy makeNewWindowManager();
public FallbackEventHandler makeNewFallbackEventHandler(Context context);
}
在实际的调用中,PolicyManager
的真正实现是Policy
类,Policy
类中的makeNewWindow
方法的实现如下,由此可以发现,Window
的具体实现的确是PhoneWindow
。
public static Window makeNewWindow(Context context) {
return new PhoneWindow(context);
}
关于策略类PolicyManager
是如何关联到Policy
上面的,这个无法从源码中的调用关系来得出,这里猜测可能是有编译环节动态控制的。到这里Window
已经创建完成了,下面分析Activity
的视图是怎么附属在Window
上的。由于Activity
的视图由setContentView
方法提供,我们只需要看setContentView
方法的实现即可:
public void setContentView(int layoutResID) {
getWindow().setContentView(layoutResID);
initWindowDecorActionBar();
}
从Activity
的setContentView
的实现可以看出,Activity
将具体实现交给了Window
处理,而Window
的具体实现是PhoneWindow
,所以只需要看PhoneWindow的相关逻辑即可。
PhoneWindow
的setContentView
方法大致遵循如下几个步骤:
1. 如果没有DecorView,那么就创建它
DecorView
是一个FrameLayout
,DecorView
是Activity
中的顶级View
,一般来说,它的内部包含标题栏和内部栏,但是这个会随着主题的变换而发生改变。不管怎么样,内容栏是一定要存在的,并且内容栏具体固定的id
,那就是content
,它的完整id
是android.R.id.content
。DecorView
的创建过程由installDecor
方法来完成,在方法内部会通过generateDecor
方法来直接创建DecorView,这个时候DecorView
还只是一个空白的FrameLayout
。
protected DecorView generateDecor(){
return new DecorView(getContext(),-1);
}
为了初始化DecorView
的结构,PhoneWindow
还需要通过generateLayout
方法来加载具体的布局文件到DecorView
中,具体的布局文件和系统版本以及主题有关。
View in = mLayoutInflater.inflate(layoutResource, null);
decor.addView(in, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));
mContentRoot = (ViewGroup) in;
ViewGroup contentParent = (ViewGroup)findViewById(ID_ANDROID_CONTENT);
其中ID_ANDROID_CONTENT
的定义如下,这个id
所对应的ViewGroup
就是mContentParent:
public static final int ID_ANDROID_CONTENT = com.android.internal.R.id.content;
2. 将View添加到DecorView的mContentParent中
由于在步骤1
中已经创建并初始化了DecorView
,因此这一步直接将Activity
的视图添加到DecorView
的mContentParent
中即可:mLayoutInflater.inflate(layoutResID, mContentParent)
。到此为止,Activity
的布局文件已经添加到DecorView
里面了,由此可以理解Activity
的setContentView
这个方法的来历了。
3. 回调Activity的onContentChanged方法通知Activity视图已经发生改变
由于Activity实现了Window的Callback接口,这里表示Activity的布局文件已经被添加到DecorView的mContentParent中了,于是需要通知Activity,使其可以做相应的处理。Activity的onContentChanged方法是个空实现,我们可以在子Activity中处理这个回调。这个过程的代码如下所示:
final callback cb = getCallback();
if(cb != null && !isDestroyed()){
cb.onContentChanged();
}
到这里为止DecorView已经被创建并初始化完毕,Activity的布局文件也已经成功添加到了DecorView的mContentParent中,但是这个时候DecorView还没有被WindowManager正式添加到Window中,这里需要正确理解Window的概念,Window更多表示的是一种抽象的功能集合,虽然说早在Activity的attach方法中Window就已经被创建了,但是这个时候由于DecorView并没有被WindowManager识别,所以这个时候的Window无法提供具体功能,因为它还无法接收外界的输入信息。在ActivityThread的handleResumeActivity方法中,首先会调用Activity的onResume方法,接着会调用Activity的makeVisible(),正是在makeVisible()方法中,DecorView真正地完成了添加和显示这两个过程,到这里Activity的视图才能被用户看到。
void makeVisible() {
if (!mWindowAdded) {
ViewManager wm = getWindowManager();
wm.addView(mDecor, getWindow().getAttributes());
mWindowAdded = true;
最后
感谢您的阅读,在文末给大家准备一个福利。本人从事Android开发已经有十余年,算是一名资深的移动开发架构师了吧。根据我的观察发现,对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。
所以在此将我十年载,从萌新小白一步步成长为Android移动开发架构师的学习笔记,从Android四大组件到手写实现一个架构设计,我都有一一的对应笔记为你讲解。
当然我也为你们整理好了百度、阿里、腾讯、字节跳动等等互联网超级大厂的历年面试真题集锦。这也是我这些年来养成的习惯,一定要学会把好的东西,归纳整理,然后系统的消化吸收,这样才能极大的提高学习效率和成长进阶。碎片、零散化的东西,我觉得最没有价值的。就好比你给我一张扑克牌,我只会觉得它是一张废纸,但如果你给我一副扑克牌,它便有了它的价值。这和我们收集资料就要收集那些系统化的,是一个道理。
最后,赠与大家一句诗,共勉!
不驰于空想,不骛于虚声。不忘初心,方得始终。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!
了它的价值。这和我们收集资料就要收集那些系统化的,是一个道理。
[外链图片转存中…(img-xDvV0lgC-1715337410385)]
最后,赠与大家一句诗,共勉!
不驰于空想,不骛于虚声。不忘初心,方得始终。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!