美文网首页
第8章 理解 Window 和 WindowManager

第8章 理解 Window 和 WindowManager

作者: SunnyGL | 来源:发表于2020-04-27 22:29 被阅读0次

    Window的添加过程

    WindowManagerImpl.addView
    -> WindowManagerGlobal.addView(检查参数,创建 ViewRootImpl,并将 View 保存在列表中)
    -> ViewRootImpl.setView (requestLayout 方法对 View 进行测量,布局,绘制)
    -> WindowSession.addToDisplay(方法中有个 mWindow 参数,为 ViewRootImpl 内部类 W 实例)
    -> Session.addToDisplay (Session为WindowSession的实现类,IPC)
    -> WindowManagerService.addWindow

    Window的删除过程

    WindowManagerImpl.removeView
    -> WindowManagerGlobal.removeView (从列表中找到待删除View的下标,然后调用 removeViewLocked 方法,在 removeViewLocked 方法内部,调用 ViewRootImpl.die 方法,最后将 待删除 View 添加到 mDyingViews 集合中)
    -> ViewRootImpl.die (异步删除,使用 Handler 发送一条消息。同步删除,直接调用 doDie 方法,doDie 内部调用 dispatchDetachedFromWindow 方法,dispatchDetachedFromWindow 方法内部主要做以下四件事)

    (1)垃圾回收相关的工作,比如清除数据和消息、移除回调。
    (2)通过Session的remove方法删除Window:mWindowSession.remove(mWindow), 这同样是一个IPC过程,最终会调用WindowManagerService的removeWindow方法。
    (3)调用View的dispatchDetachedFromWindow方法,在内部会调用View的 onDetached-FromWindow()以及onDetachedFromWindowInternal()。对于 onDetachedFromWindow()大家一定不陌生,当View从Window中移除时,这个方法就会被 调用,可以在这个方法内部做一些资源回收的工作,比如终止动画、停止线程等。
    (4)调用WindowManagerGlobal的doRemoveView方法刷新数据,包括mRoots、 mParams以及mDyingViews,需要将当前Window所关联的这三类对象从列表中删除。

    Window的更新过程

    WindowManagerImpl.updateViewLayout (更新View的LayoutParams,替换列表中老的LayoutParams,再更新ViewRootImpl中的LayoutParams)
    -> ViewRootImpl.setLayoutParams (内部调用 scheduleTraversals 方法对 View 重新布局,包括测量、布局、重绘)
    -> WindowSession来更新Window的视图 (IPC)
    -> WindowManagerService.relayoutWindow()

    总体流程

    WindowManager -> ViewRoot -> IPC 到 WindowManagerService


    Window 的添加、删除更新过程.png

    Activity Window 创建过程

    Activity.attach(创建 PhoneWindow)

    Activity.setContentView
    -> PhoneWindow.setContentView(如果 DecorView 为空,创建 DecorView,之后给 DecorView 设置基础布局,拿出 content view,将 Activity 内容布局加载到 content view 上,最后回调
    Activity 的 onContentChanged 方法)

    ActivityThread.handleResumeActivity
    -> Activity.onResume - Activity.makeVisible
    -> WindowManager.addView(IPC)


    Activity Window 创建过程.png

    Dialog的Window创建过程

    Dialog 构造方法(创建 PhoneWindow)
    Dialog.setContentView(PhoneWindow.setContentView)
    Dialog.show
    -> WindowManager.addView(IPC)


    Dialog 的 Window 创建过程.png

    Toast的Window创建过程

    Toast.show(参数tn:自身用于和 NMS 通信的Binder)
    -> NMS.enqueueToast(封装为 ToastRecord,将 ToastRecord 添加到列表中,调用 showNextToastLocked 方法,在其内部调用 tn 的 show 方法,并开启计时,计时结束后调用 tn 的 hide 方法)
    -> Toast.TN.show(通过 Handler 从 Binder 线程池回到调用线程,接着获取系统 WindowManager,调用 WindowManager 的 addView)
    -> Toast.TN.hide(通过 Handler 从 Binder 线程池回到调用线程,接着调用 WindowManager 的 removeViewImmediate 方法)


    Toast 的 Window 创建过程.png

    关于 Activity 与 Window、PhoneWindow、DecorView 的关系,这篇文章分析的极佳。
    https://zhuanlan.zhihu.com/p/26834562

    相关文章

      网友评论

          本文标题:第8章 理解 Window 和 WindowManager

          本文链接:https://www.haomeiwen.com/subject/hzupwhtx.html