正常大家都清楚,当我们view发生一些必须要重新测量才能生效的属性的时候,我们会尝试触发view.requestLayout从而让整个view树发生重新测量与布局。
今天的问题是:
- 假设requestLayout的调用发生在灭屏/切home之后会立即触发吗?
- 如果1不会立即触发,当我们回到应用,会自动重新触发一次requestLayout吗?
更多问答 >>
-
每日一问 | 已经有了 Intent,那为啥还要 PendingIntent?
2021-05-28 00:29 -
2021-05-28 00:29
-
每日一问 | Dialog 的构造方法的 context 必须传入 Activity吗?
2021-07-11 22:06 -
2021-07-11 22:06
-
每日一问 | 我们经常说到的 Android 脱糖指的是什么?
2021-07-11 22:06 -
2021-05-06 00:16
-
每日一问 | 听说你做过内存优化 之 Bitmap内存占用到底在哪?
2021-04-19 23:40 -
2021-04-08 00:25
-
每日一问 | mipmap vs drawable,傻傻分不清楚?
2021-03-30 21:14 -
每日一问 | onDraw 里面调用 invalidate 做动画,有什么问题?
2021-04-13 00:31
1. 背景
最近有个粉丝跟我提了一个很有深度的问题。
乍一看有点超纲了。细细一想,我把这个问题拆分成了两个问题,本文我将紧紧围绕这两个问题,讲解requestLayout背后的故事。
postSyncBarrier我知道,Handler的同步屏障机制嘛,但是锁屏之后为什么还要调用requestLayout()呢?于是我脑补了一个场景。
我脑补的这个场景,用罗翔老师的话来讲是 “法律允许,但是不提倡”。当Activity不在前台的时候,就应该把requestLayout()方法停掉嘛,我们知道的,这个方法会从调用的View一层一层往上调用直到ViewRootImpl.requestLayout()方法,然后会从上往下触发View的测量和布局甚至绘制方法。非常之浪费嘛!错误非常之低级!但是果真如此吗?
电竞主播芜湖大司马,有一句网络流行语你以为我在第一层,其实我在第十层。下面我将用层级来表示对requestLayout方法的了解程度,层级越高,表示了解越深刻。
了解我的粉丝都知道,我喜欢用树形图来分析Android View源码。上图:
2. 第一层(往上,层层遍历)
假设调用I.requestLayout(),会触发哪些View的requestLayout方法?
答:会依次触发I.requestLayout() -> C.requestLayout() -> A.requestLayout() -> ...省略一些View -> ViewRootImpl.requestLayout()
该方法作用如下:
重点看下mParent.isLayoutRequested()方法,它在View.java中有具体实现
如果mPrivateFlags增加PFLAG_FORCE_LAYOUT标志位,则认为View已经请求过布局。由前文可知,在requestLayout的第二步会增加该标志位。熟悉位操作的朋友就会知道,有增加操作就会有对应的清除操作。 经过一番搜索,找到:
在View调用完layout方法,会将PFLAG_FORCE_LAYOUT标志位清除掉。当View下次再调用requestLayout方法时,依旧能往上层层调用。但是如果当layout()方法没有执行时,下次再调用requestLayout方法时,就不会往上层层调用了。
所以先回答文章开始的第一个问题:
如果你知道requestLayout调用是一个层级调用,那么恭喜你,你已经处于认知的第一层了。送你一张二层入场券。
3. 第二层(ViewRootImpl.requestLayout)
我们来看看第一层讲到的ViewRootImpl.requestLayout()
该方法主要作用如下:
此处有三个特别重要的知识点:
mTraversalRunnable相对比较简单,它的作用就是从ViewRootImpl 从上往下执行performMeasure、performLayout、performDraw。[重点:敲黑板]它的执行时机是当Vsync信号来到时,会往主线程的Handler对应的MessageQueue中发送一条异步消息,由于在scheduleTraversals()中给MessageQueue中发送过一条同步屏障消息,那么当执行到同步屏障消息时,会将异步消息取出执行
4. 第三层(TraversalRunnable)
当vsync信号量到达时,Choreographer会发送一个异步消息。当异步消息执行时,会调用ViewRootImpl.mTraversalRunnable回调。
它的作用:
performTraversals()方法特别复杂,给出伪代码如下
该方法的作用:
mStopped表示Activity是否处于stopped状态。如果Activity调用了onStop方法,performLayout方法是不会调用的。
回答文章开始的第二个问题:
至此第一层里面留下的小悬念也得以解开,因为不会执行View.layout()方法,所以PFLAG_FORCE_LAYOUT不会被清除,导致接下来的requestLayout方法不会层层往上调用。
至此本文的两个问题都已经得到了答案。
当我把问题提交给鸿洋大佬的wanandroid上时,大佬又给我提了一个问题。
于是我写了个demo来验证
在自定义布局的onLayout方法中打印日志
锁屏,1s后调用requestLayout,日志没有打印,1s后亮屏,发现日志打印了。
所以
有了demo找原因就很简单了。正面不好攻破,那就祭出调试大法呗。但是断点放在哪好呢?思考了一番。我觉得断点放在发送同步屏障的地方比较好,ViewRootImpl.scheduleTraversals()。为什么断点放这里?(那你就得了解同步屏障和vsync刷新机制了,后文会讲)
亮屏后,发现断点执行了。从堆栈中可以看出Activity的performRestart()方法执行了ViewRootImpl的scheduleTraversals方法。
虽然,亮屏的时候没有执行View.requestLayout方法,由于锁屏后1s执行了View.requestLayout方法,所以PFLAG_FORCE_LAYOUT标记位还是有的。亮屏调用了performTraversals方法时,会执行Measure、Layout、Draw等操作。至此,完美回答了粉丝和鸿洋大佬的问题
5. 第四层(Handler同步屏障)
Handler原理,也是面试必问的问题。涉及到很多知识点。线程、Looper、MessageQueue、ThreadLocal、链表、底层等技术。本文我就不展开讲了。如果对Handler不是很了解。也不影响本层次的学习。但是还是强烈建议看完本文后再另行补课。
msg5出队列
此刻msg2并不会出队列,队列中已经没有了红色消息,但是存在黄色消息,所以会一直等红色消息,绿色消息得不到执行机会
调用removeSyncBarrier()方法,将msg1出队列
绿色消息按顺序出队
同步屏障就介绍到这,如果没明白的话,建议网上搜索其它资料阅读。
6. 第五层(Choreographer vsync机制)
}
```7. 第六层(绘制机制)
ViewRootImpl和Choreographer是绘制机制的两大主角。他们负责功能如下。具体就不展开写了。
好棒啊👍答的很全面~
多谢。久仰
妈妈再也不担心我面试被问到绘制流程了
有点不明白为什么锁屏1s后会再执行resquestLayout
人家那是自己模拟的场景,不是系统规则。
大佬,NB
好问题!一个问题囊括了众多重要知识点。
首先问题 1 的答案是会触发,但不会执行重新布局流程,这里要了解 requestLayout() 函数,方法会一次向上给回溯父控件,标记沿途父控件需要重新 Layout、视图无效需要重新绘制。最终到达 ViewRootImpl 中,执行 scheduleTraversals() 函数,进而触发 performTraversals() 流程,在该流程中会对当前 App 可见性和与ViewRootImpl 对应 DecorView 可见性作出判断,不会执行三大流程。
问题 2 的答案是会重新触发测量、布局、绘制过程,但是这一次并不对应上一次的 requestlayout() 请求,这一次仅仅是当 App 重新显示时,位于栈顶的 Activity 对应的 ViewRootImpl 会执行一次 scheduleTraversals() 进而进行一次测量、布局、绘制过程。由于上一次的 requestLayout() 已经标记了视图无效需要重新 layout() 的控件,在三大步骤中会对需要重新布局的控件进操作。