Window 的添加过程

本文深入探讨了Android系统中Window的添加过程,从WindowManager的addView方法入手,详细介绍了如何通过ViewRootImpl的setView方法更新界面,并通过IPC调用WindowManagerService的addWindow方法完成Window的添加。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Window 的添加过程

Window(或者说View) 是怎么添加到 Android 系统中然后展示给用户的?让我们来探索一下 Window 的添加过程。

Window 添加过程的入口方法

要探索添加的过程,必须先在源代码中找到添加 Window 的入口方法。

Window 的添加需要通过 WindowManager 的 addView 方法实现,但 WindowManager 是个接口,它的真正实现类是 WindowManagerImpl 类,但 WindowManagerImpl 也并没有直接实现对 Window 的添加、删除、更新操作,而是通过桥接模式将所有操作委托给 WindowManagerGlobal 去实现。最终会调用 WindowManagerGlobal 类的 addView 方法真正开启 View 的添加过程。

所有,Window 添加过程的真正入口方法实际上是 WindowManagerGlobal 类的 addView 方法。

Window 添加过程的主要流程

WindowManagerGlobal 的 addView 方法主要干了三件事:

  1. 检查参数 params 是否是 WindowManager.LayoutParams,如果不是说明参数不合法,则会抛出异常。

    public void addView(View view, ViewGroup.LayoutParams params,
            Display display, Window parentWindow) {
        if (view == null) {
            throw new IllegalArgumentException("view must not be null");
        }
        if (display == null) {
            throw new IllegalArgumentException("display must not be null");
        }
        if (!(params instanceof WindowManager.LayoutParams)) { // 检查 params 参数是否合法
            throw new IllegalArgumentException("Params must be WindowManager.LayoutParams");
        }
        ...
    }
    
  2. 创建 ViewRootImpl,并将 View 添加到列表中。

    public void addView(View view, ViewGroup.LayoutParams params,
            Display display, Window parentWindow) {
            ...
    		root = new ViewRootImpl(view.getContext(), display); // 创建 ViewRootImpl
            view.setLayoutParams(wparams);
            mViews.add(view); // 将View添加到mView列表中,mView 存储的是所有Window对应的View
            mRoots.add(root);
            mParams.add(wparams);
            ...
    }
    
  3. 通过 ViewRootImpl 的 setView 方法来添加更新界面并通过 IPC 的方式调用 WindowManagerService 的 addWindow 方法完成 Window 的添加过程。

    public void addView(View view, ViewGroup.LayoutParams params,
    Display display, Window parentWindow) {
    		...
            // do this last because it fires off messages to start doing things
            try { 
                root.setView(view, wparams, panelParentView); // ViewRootImpl的setView 方法
            } catch (RuntimeException e) {
                // BadTokenException or InvalidDisplayException, clean up.
                synchronized (mLock) {
                    final int index = findViewLocked(view, false);
                    if (index >= 0) {
                        removeViewLocked(index, true);
                    }
                }
                throw e;
            }
    }
    
    • ViewRootImpl 的setView 方法是如何实现界面的更新的呢?

      setView 方法中会调用 requestLayout() 方法去完成异步刷新请求:

      @SuppressWarnings({"EmptyCatchBlock", "PointlessBooleanExpression"})
      public final class ViewRootImpl implements ViewParent,
              View.AttachInfo.Callbacks, HardwareRenderer.HardwareDrawCallbacks {
          private static final String TAG = "ViewRootImpl";
      	...
          // Schedule the first layout -before- adding to the window
          // manager, to make sure we do the relayout before receiving
          // any other events from the system.
          requestLayout();           
      } 
      

      我们再查看 requestLayout 方法的源码,看它干了什么:

      @Override
      public void requestLayout() {
          if (!mHandlingLayoutInLayoutRequest) {
               checkThread();
               mLayoutRequested = true;
               scheduleTraversals(); // scheduleTraversals 方法是View绘制的入口
          }
      }
      

      可以看到,是调用了 scheduleTraversals 方法进行绘制,scheduleTraversals 方法最终会调用 performTraversals 方法,我们知道 performTraversals 是 View 执行绘制过程的入口方法,该方法会经过测量、布局、绘制这三个过程把 View 绘制出来。

    • View 绘制出来以后是怎么通过IPC调用的方式添加到 Window 中的呢?

      我们知道,WindowManager 是外界访问 Window 的入口,所以最终 WindowManager 会通过 IPC 的方式调用 WindowManagerService 的 addWindow 方法,这样一来, Window 的添加请求就交给了 WindowManagerService 来处理了,然后 WindowManagerService 会经过一系列的操作将 View 添加到 Window 中并展示出来。

      作为应用层开发者来说,了解到这个程度个人觉得就可以了,没必要去深究 WindowManagerService 的实现细节,至于 WindowManagerService 是如何处理 Window 的添加请求的,感兴趣的读者可以去查看源码。

      参考书籍:《Android 开发艺术探索》

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

老陈Android馆

你的赞赏将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值