View 的测量过程中,有一个比较重要的类需要掌握:MeasureSpec。我们在阅读源码的时候会发现,在 View 的测量过程中,MeasureSpec 是一个会经常出现的类,如果不先掌握这个类的话,是没法阅读下去的。
MeasureSpec 会在很大程度上决定一个 View 的尺寸规格,之所以是很大程度上是因为这个过程还受父容器的影响,因为父容器影响 View 的 MesaureSpec 的创建过程。
在测量过程中,系统会将 View 的 LayoutParams 根据父容器所施加的规则转换成对应的 MeasureSpec,然后再根据这个 MeasureSpec 来测量出 View 的宽/高。
MeasureSpec
MeasureSpec 是一个int值,但是这个int值被分为了两部分,一部分表示 SpecMode (测量模式),一部分表示 SpecSize(在某种测量模式下的大小)。
可能有很多人想不通,一个int型整数怎么可以表示两个东西(大小模式和大小的值),一个int类型我们知道有32位。而模式有三种,要表示三种状态,至少得2位二进制位。于是系统采用了最高的2位表示模式。如图:
最高两位是00的时候表示"未指定模式",即MeasureSpec.UNSPECIFIED。
最高两位是01的时候表示"'精确模式",即MeasureSpec.EXACTLY。
最高两位是10的时候表示"最大模式",即MeasureSpec.AT_MOST。
-
精确模式(MeasureSpec.EXACTLY)
在这种模式下,尺寸的值是多少,那么这个组件的长或宽就是多少。
-
最大模式(MeasureSpec.AT_MOST)
这个也就是父组件,能够给出的最大的空间,当前组件的长或宽最大只能为这么大,当然也可以比这个小。
-
未指定模式(MeasureSpec.UNSPECIFIED)
这个就是说,当前组件,可以随便用空间,不受限制。
MeasureSpec 通过将 SpecMode 与 SpecSize 打包成一个 int 值来避免过多的对象内存分配。为了方便操作,它还提供了对应的打包与解包方法。
// 将 size 与 mode 组合成一个 MeasureSpec 对象
android.view.View.MeasureSpec#makeMeasureSpec
// 从 MeasureSpec 中获取 mode
android.view.View.MeasureSpec#getMode
// 从 MeasureSpec 中获取 size
android.view.View.MeasureSpec#getSize
在 View 测量的时候,系统会将 LayoutParams 在父容器的约束下转换成对应的 MeasureSpec,然后再根据这个 MeasureSpec 来确定 View 测量后的大小。(这里需要注意,MeasureSpec 不是由 LayoutParams 唯一决定的,而是由 LayoutParams 与父布局一起决定的)
各种 View 测量的区别
顶层 View
我们知道一般的 View 都会有父布局,但是最顶层的 View 是没有的,那么它是如何测量的呢?
首先它会获取 LayoutParams,再判断 LayoutParams 宽高的值:
- 如果为 LayoutParams.MATCH_PARENT,这表示精确模式,大小就是窗口大小。
- 如果为 LayoutParams.WRAP_CONTENT,这表示最大模式,大小未定,但是不能超过窗口大小。
这就比较简单了,顶层View的测量,一般宽高都是 LayoutParams.MATCH_PARENT,大小为窗口大小。
子 View
对于普通的 View 来说,它的测量与父布局有关,而每个父布局的特性又不同,无法每个都涉及到,所以这里采取一个“管中窥豹,可见一斑”的方法。
这里介绍一下 ViewGroup 的 measureChildWithMargins 方法。
android.view.ViewGroup#measureChildWithMargins
protected void measureChildWithMargins(View child,
int parentWidthMeasureSpec,