7.Activity从创建到显示重要角色介绍

所有代码都是Android 11

根据Activity 中各个出现的角色的顺序,来了解Activity 从创建到显示的流程

1.PhoneWindow -->窗口的持有者,每个activity都需要持有一个window,通过ISessionService 与 WMS 交互, 将初始化后的surface 与 SurfaceFliger 中的 QueueByte 关联,PhonwWindow 创建时机 activity 反射创建之后, 在Activity.attach 方法中创建,

 ActivityThread.java -->performLaunchActivity   先通过mInstrumentation 反射创建Activity ,在Activity.attach 中创建

2.DecorView -->Activity.setContentView 中所有View 的根节点,创建时机 PhoneWindow.setContentView, setContentView 实际调用的是getWindow.getContentView(),

Activity.java.onCreate() --->   Activity.java.setContentView() -->PhoneWindow.setContentView --> 创建 DecorView   ,解析所有View,并添加到DecorView   中

在这里会出现几个经典面试题
经典面试题

  1. setContentView后,这里能获取到View 的宽高吗,为什么
    2.setContentView后,此时View关联到了activity 之上了吗,为什么
    3.setContentView后,这里可以可以使用线程更新View吗,为什么 ,
    带着这3个疑问,我们来继续分析

3.WindowManager -->在 ActivityThread 收到 AMS 执行Activity onResume 的 handleResumeActivity 方法中,先执行Activity 的 onResume 方法, 再通过WindowManager.addView 方法,这里调用的是WindowManagerImpl.add 方法 ,将 DecorView 与 PhoneWindow 做关联

为什么调用的是WindowManagerImpl,Activity.attach 初始化的 mWindowManager 是通过Window 来创建的,而Window 创建 WindowMananger 的代码如下

Window.java

    public void setWindowManager(WindowManager wm, IBinder appToken, String appName,
            boolean hardwareAccelerated) {
        mAppToken = appToken;
        mAppName = appName;
        mHardwareAccelerated = hardwareAccelerated;
        if (wm == null) {
            wm = (WindowManager)mContext.getSystemService(Context.WINDOW_SERVICE);
        }
        mWindowManager = ((WindowManagerImpl)wm).createLocalWindowManager(this);
    }

可以看到Window 创建WindowManager 的是调用 WindowManagerImpl.createLocalWindowManager方法来创建的,继续跟踪如下

WindowManagerImpl.java

  public WindowManagerImpl createLocalWindowManager(Window parentWindow) {
      return new WindowManagerImpl(mContext, parentWindow);
  }

4.WindowManagerGlobal -->单利创建,整个app进程只有一个 ,他的主要作用就是管理着所有的View ,所有的ViewRootImlp , 以及所有Window.LayoutParams

我们上面分析到WindowManagerImpl.addView 方法,他这里面2行代码,最终的一行就是调用 WindowManagerGlobal.addView

5.ViewRootImpl -->在 WindowManagerGlobal.addView 创建ViewRootImpl ,并将 DecorView 放入到ViewRootImpl 中, 在ViewRootImpl 方法中,先调用了requestLayout 方法, requestLayout 方法的调用逻辑是先checkThread ,然后发送消息屏障,在发送一个刷新View 的回调,等待垂直同步信号来临后,由mChoreographer 处理各种事件后,在调用doTraversal移除屏障消息,处理刷新事件,执行View树种各个View 的 onMeasure onLayou onDraw 等事件

看到这里我们就能回答上面那三个经典的面试题了,

1.setContentView后,这里能获取到View 的宽高吗
答:不能,在setContentView 后,只是将布局View 与 DecorView 做了关联,并没有执行View 的测量 流程,真正的测量流程是在 Activity 的 onResume 之后的下一个requestLayout 发出的屏障消息之后,

2.setContentView后,此时View关联到了activity 之上了吗
答:没有 在setContentView 后,只是将布局View 与 DecorView 做了关联,实际关联的地方再 Activity 的 onResume之后 调用的 WindowManager.addView ,通过WindowManagerGlobal 创建ViewRootImpl ,通过 ViewRootImpl 将View 与 ViewRootImpl Window 做的关联

3.setContentView后,这里可以可以使用线程更新View吗,还有在子线程更新UI 的方法吗,
答:可以 ,之所以不能在子线程中更新UI 的原因是因为 ViewRootImpl 的 刷新方法中调用了 checkThread 方法,来判断的当前线程,而 ViewRootImpl 的创建时机是在 Activity 的 onResume 方法之后,所以在 onCreate 方法开启线程由于 ViewRootImpl 都没有创建,所以不会触发这个判断,
想要在子线程更新UI ,我们先来分析一下ViewRootImpl 的checkThread 方法


  public ViewRootImpl(Context context, Display display, IWindowSession session,
          boolean useSfChoreographer) {
      mThread = Thread.currentThread();
}

  void checkThread() {
      if (mThread != Thread.currentThread()) {
          throw new CalledFromWrongThreadException(
                  "Only the original thread that created a view hierarchy can touch its views.");
      }
  }

从上面可以看到 mThread 就是创建 ViewRootImpl 的线程,那么在更新时候的线程如果不是 ViewRootImpl 的创建线程就会报错,只要我们在子线程中创建ViewRootImpl ,就可以做到在子线程中刷新UI

6. 重中之重 Furface -->上面的流程执行完毕后, 需要准备的Window , View 树 都准备好了, 那么接下来就要开始绘制View 了,我们都知道在View 的onDraw 方法中有一个参数就是canvas ,那么这个canvas 是怎么来的呢,

ViewRootImpl.java -->performTraversals() 方法
可能很多人比较关注的是 performTraversals方法中调用到了 View 的 performMeasure performLayout performDraw,但是在测量与绘制之前,更重要的一部则是创建画布,如果连画布都没有,创建的View 也是没有意义的,而创建好了画布,又需要将画布上的内容交给显示器去显示,这也是需要在 测量与绘制之前就需要准备好,这里就涉及到了跨进程通信的流程

--> 创建Suface

ViewRootImpl.performTraversals() 
    --> ViewRootImpl.relayoutWindow()//测量Window 的大小,并将 mSurfaceControl 通过 ISessionManager 交给WMS关联SurfaceFliger
            -->Session.relayout()
                  -->WindowManagerService.relayoutWindow()

在来分析一下 WindowManagerService的 relayoutWindow方法

WindowManagerService.relayoutWindow()
    -->WindowManagerService.createSurfaceControl()
            -->   WindowStateAnimator.createSurfaceLocked()
                    -->WindowSurfaceController()
                              -->SurfaceControl.Builder.build()
                                      -->SurfaceControl()
                                                -->nativeCreate()

到了这里参与绘制的所有参与者都已经初始化完毕,接下来就可以执行 测量 布局 绘制等流程了,而这里还有一个比较重要的地方就是view 的canvas 是如何来的,

canvas = mSurface.lockCanvas(dirty);

DecorView 的canvas 是通过lockCanvas 来获取,而他的canvas 会一级一级向下传递,也就是说DecorView 上所有的子View 会公用一个surface 上个canvas,也就是同一个surface

这里也有一个经典面试题,那就是SurfaceView 与普通的View 有什么不同,他的不同为他带来了哪些好处,哪些不好之处,简要说一下SurfaceView
这里我们继续带着问题来分析接下来的流程
个人理解: SurfaceView 同样是一个View , 他与其他View 最大不同之处就在于,SurfaceView会单独持有一个 surface ,可以利用线程来更新View ,更大程度的保证在复杂的场景的刷新帧率,而他所有的操作都是借助 SurfaceHolder 来完成的,相对其他View 的操作性并不好,

经典面试题: View Window WindowManager WindowManagerService 的关系是什么样的

个人理解:Window 是 Activity 的显示的窗口, Activity 通过 WindowManager 管理这所有 Window , 并且WindowManager 通过 WindowManagerGlobal 来管理这所有ViewRootImpl 与 View , WindowManager 为所有View 的显示完善了 SurfaceContral ,并将 Surface 与 SurfaceFliger 的 QueueByte 关联起来,为显示器提供显示数据

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,185评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,445评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,684评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,564评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,681评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,874评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,025评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,761评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,217评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,545评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,694评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,351评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,988评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,778评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,007评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,427评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,580评论 2 349