面试复习题--Window 与 WMS 绑定的详细流程

握 Android 中 Window 与 WMS(WindowManagerService)完成绑定的全链路详细流程,包括从应用层触发绑定、跨进程通信到 WMS 内部处理的每一个核心环节 —— 这是理解 Window 如何从「内存对象」变为「屏幕可见界面」的核心,整个流程跨越应用进程和 SystemServer 进程,核心依赖 Binder 通信和 ViewRootImpl 的桥梁作用。下面我会以应用 Activity 的 Window(PhoneWindow) 为例,拆解从触发绑定到最终完成的完整步骤,并结合源码级逻辑说明关键细节。


一、核心前提:绑定流程的参与角色

先明确全流程的核心参与者,理解角色分工才能理清流程逻辑:

角色(进程) 核心类 / 接口 作用
应用进程 1. WindowManagerImpl/WindowManagerGlobal(WindowManager 的实现)2. ViewRootImpl(应用 - WMS 桥梁)3. PhoneWindow(Window 具体实现)4. DecorView(Window 的顶级 View) 触发绑定请求、创建通信桥梁、承载 UI 内容
SystemServer 进程 1. WMS(WindowManagerService)2. IWindowSession(Binder 接口,WMS 的会话)3. WindowToken(Window 身份凭证)4. SurfaceControl(Surface 管理) 接收绑定请求、校验 Window 合法性、分配 Surface、管理 Window 生命周期
通信桥梁 IWindow/IWindowSession(AIDL 生成的 Binder 接口) 应用进程与 WMS 的跨进程通信载体(ViewRootImpl 实现 IWindow,WMS 提供 IWindowSession)

二、Window 与 WMS 绑定的完整流程(分 6 个核心阶段)

整个流程的核心入口是WindowManager.addView(),最终以 WMS 完成 Window 注册并分配 Surface 为绑定完成标志,流程可视化如下:

生成失败,请重试

阶段 1:前置准备 ——Window 与 WindowManager 初始化(应用进程)

这是绑定前的基础工作,Activity 启动时已完成:

  1. Activity.attach():创建PhoneWindow实例(Activity 的专属 Window),并通过mWindow = new PhoneWindow(this)完成绑定;
  2. PhoneWindow 初始化 WindowManager:PhoneWindow 通过getWindowManager()返回WindowManagerImpl实例(WindowManager 的具体实现);
  3. setContentView():给 PhoneWindow 的DecorView加载开发者编写的布局,此时 DecorView 仅完成初始化,未与 WMS 绑定。

核心代码(简化)

java

运行

// Activity.java
final void attach(...) {
    mWindow = new PhoneWindow(this); // 创建PhoneWindow
    mWindowManager = mWindow.getWindowManager(); // 获取WindowManagerImpl
}

// PhoneWindow.java
public WindowManager getWindowManager() {
    return mWindowManager != null ? mWindowManager : (mWindowManager = createLocalWindowManager());
}
private WindowManager createLocalWindowManager() {
    return new WindowManagerImpl(mContext); // 初始化WindowManagerImpl
}

阶段 2:触发绑定 ——WindowManager.addView ()(应用进程)

Activity 执行onResume()后,系统会触发WindowManager.addView(),这是绑定流程的「启动开关」:

  • WindowManager是「门面类」,真正的逻辑在WindowManagerGlobal(全局单例,管理所有应用 Window)中;
  • 核心操作:将 DecorView、LayoutParams 传入,创建ViewRootImpl(应用与 WMS 的核心桥梁)。

核心代码(简化)

java

运行

// WindowManagerImpl.java(门面类,转发请求)
@Override
public void addView(@NonNull View view, @NonNull ViewGroup.LayoutParams params) {
    applyDefaultToken(params);
    // 转发给WindowManagerGlobal处理
    mGlobal.addView(view, params, mContext.getDisplay(), mParentWindow);
}

// WindowManagerGlobal.java(核心处理类)
public void addView(View view, ViewGroup.LayoutParams params, ...) {
    // 1. 参数校验:LayoutParams必须是WindowManager.LayoutParams
    final WindowManager.LayoutParams wparams = (WindowManager.LayoutParams) params;
    
    // 2. 关键:为当前Window创建ViewRootImpl(每个Window对应一个ViewRootImpl)
    ViewRootImpl root = new ViewRootImpl(view.getContext(), display);
    
    // 3. 关联View、LayoutParams、ViewRootImpl
    view.setLayoutParams(wparams);
    mViews.add(view); // 存储所有Window的View
    mRoots.add(root); // 存储所有ViewRootImpl
    mParams.add(wparams); // 存储所有LayoutParams
    
    try {
        // 4. 调用ViewRootImpl.setView(),真正触发与WMS的绑定
        root.setView(view, wparams, panelParentView);
    } catch (RuntimeException e) {
        // 异常处理:移除已添加的View/ViewRootImpl
    }
}

阶段 3:通信准备 ——ViewRootImpl.setView ()(应用进程)

ViewRootImpl.setView()是应用进程内的核心准备步骤,完成后将发起跨进程请求:

  1. 关联 DecorView:将 DecorView 赋值给 ViewRootImpl 的mView成员,作为 Window 的顶级 Vi
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值