PhoneWindow,ViewRoot,Activity之间的大致关系

本文深入探讨了Android中Activity与Window的关系,解释了PhoneWindow的作用及其内部实现机制,包括UI组件的构建过程及事件分发原理。
android里,我们都知道activity.但是一个activity跟一个Window是一个什么关系呢?

activity.java中,我们可以看到两个变量,分别是:

private Window mWindow;

private WindowManager mWindowManager;

这这个变量是在哪里赋值的呢?可以看到attach函数,

mWindow = PolicyManager.makeNewWindow(this);

跟进去看一下,可以发现返回的时候一个PhoneWindow对象,PhoneWindowWindow类的派生类。

mWindow.setWindowManager(null, mToken, mComponent.flattenToString());

if (mParent != null) {

mWindow.setContainer(mParent.getWindow());

}

mWindowManager = mWindow.getWindowManager();

仔细跟踪下这段代码中我们可以得到WindowManager对象,并且这个对象是在系统唯一的,这个对象同样被赋值给PhoneWindow的成员变量。

我们在新建一个activity的时候,往往喜欢用如下的代码来定义该activityUI界面,那么这个具体的实现是怎么来实现的呢?

public void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.main);

我们跟踪setContentView就知道了。可以发现activity其实把这个操作交给了自己的mWindow来完成,通过跟踪PhoneWindow中的setContentView函数可以得出以下知识:

1,每一个PhoneWiondow中都有一个叫DecorView的对象,该对象是该PhoneWiondow的框架view,可以找到这么一段代码:

View in = mLayoutInflater.inflate(layoutResource, null);

decor.addView(in, new ViewGroup.LayoutParams(FILL_PARENT, FILL_PARENT));

layoutResource则是资源ID,通过往上翻代码就可以找得到。

2,一个PhoneWiondow中都有一个叫FrameLayout的对象mContentParent,该对象是从上述layoutResource中的布局xml中获得的,这也将是activity中所有用户新增加view都会被包含在这个对象当中。

总体可以这样大致说明:一个activity包含有一个PhoneWiondow对象,而所有的UI部件都是放在PhoneWiondow中。

ViewRoot这个类在androidUI结构中扮演的是一个中间者的角色,连接的是PhoneWindowWindowManagerService.

WindowManagerService中我们知道它读取android系统里所有事件,键盘事件,轨迹球事件等等,它怎么分发到各个activity的呢?就是通过这个ViewRoot

在每个PhoneWindow创建的时候,系统都会向WindowManger中的一个保存View数组增加PhoneWindowDecorView对象,WindowManger在保存好这个View对象的同时,也会新创建一个ViewRoot对象用来沟通WindowManagerService

可以查看WindowManagerImplViewRoot中的代码,同时ViewRoot中我们可以看到一个W类型,该类型派生自IWindow.stub可以知道这个可以被用来作远程调用。

res = sWindowSession.add(mWindow, mWindowAttributes,

getHostVisibility(), mAttachInfo.mContentInsets);

IwindowSession则是WindowManagerService中的远程调用接口,上述代码也表明了每创建了一个PhoneWindow,也就创建了一个ViewRoot,并将在WindowManagerService注册。

接下来简单的看下KeyEvent的传递。在WindowManagerService

focus.mClient.dispatchKey(event);

这个段代码的mClient则就是ViewRoot中的W类型,通过远程调用,可以看看ViewRoot的处理:

boolean handled = mView != null

? mView.dispatchKeyEventPreIme(event) : true;

mView则是PhoneWindowDecorView对象,接下来怎么具体传递可以慢慢去跟了





### ActivityWindowViewRootImpl 的关系Android 系统中,ActivityWindowViewRootImpl 是 UI 展现的核心组件,它们之间关系紧密且层次分明。以下是对其关系的详细分析: #### 1. ActivityWindow关系 每个 Activity 都对应一个 Window 对象[^3]。具体来说,Activity 中会创建一个 PhoneWindow 实例(PhoneWindowWindow 接口的唯一实现类),这个实例负责管理 Activity 的窗口内容和装饰视图(DecorView)。通过 `setContentView` 方法,Activity 将布局文件设置到 PhoneWindow 的 DecorView 中[^1]。 #### 2. Window 与 DecorView 的关系 Window 通过其内部的 DecorView 来承载所有的 UI 元素。DecorView 是整个 View 树的根节点,它是一个 FrameLayout 容器,包含了标题栏(如果存在)和内容区域(ContentArea)。当调用 `setContentView` 时,实际是将传入的布局文件添加到 DecorView 的内容区域中[^4]。 #### 3. ViewRootImpl 的角色 ViewRootImpl 是连接 Window 和 View 树的重要桥梁。当 Activity 调用 `setContentView` 后,最终会通过 WindowManager 的 `addView` 方法将 DecorView 添加到屏幕上。在此过程中,系统会创建一个 ViewRootImpl 实例,并将其与 DecorView 关联起来[^2]。以下是关键代码片段: ```java // WindowManagerGlobal 的 addView 方法 public void addView(View view, ViewGroup.LayoutParams params, Display display, Window parentWindow) { final WindowManager.LayoutParams wparams = (WindowManager.LayoutParams) params; synchronized (mLock) { ViewRootImpl root; // 创建一个 ViewRootImpl 实例 root = new ViewRootImpl(view.getContext(), display); view.setLayoutParams(wparams); mViews.add(view); // 记录所有添加的 View mRoots.add(root); // 记录所有 ViewRootImpl 实例 mParams.add(wparams); // 记录所有 LayoutParams } try { root.setView(view, wparams, panelParentView); // 将 DecorView 交给 ViewRootImpl } catch (RuntimeException e) { // 异常处理 } } ``` 从上述代码可以看出,ViewRootImpl 在初始化后,会通过 `setView` 方法接收 DecorView,并完成后续的测量、布局和绘制操作[^4]。 #### 4. 总结 - 每个 Activity 对应一个 Window,而 Window 又通过 DecorView 承载所有的 UI 元素。 - 当调用 `setContentView` 时,实际是将布局文件插入到 DecorView 的内容区域中。 - WindowManager 的 `addView` 方法会创建一个 ViewRootImpl 实例,并将 DecorView 交由 ViewRootImpl 管理,从而完成 View 树的构建和屏幕绘制过程[^3]。 ### 示例代码 以下是一个简单的示例,展示如何通过 `setContentView` 将布局文件添加到 Activity 中: ```java @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 设置布局文件 } ``` 在上述代码中,`setContentView` 方法最终会触发 Window 的 `setContentView` 方法,进而将布局文件插入到 DecorView 的内容区域中[^1]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值