众所周知,自定义 ViewGroup 中这几个方法非常重要:onMeasure, onLayout。初学者学习自定义 View 时,想必对 onMeasure 比较困惑,onMeasure是什么,为什么要测量,怎么测量?
1. onMeasure测量是什么? 为什么需要测量呢
测量的定义:确定实际尺寸。
为什么需要测量: Android 中, View 的 layout_width layout_height 并不是固定值,它可以设置成 dp、MATCH_PARENT、WRAP_CONTENT 三种形式。
1.我们试想,如果用户设置了精确值 dp,那就好说,直接给 View 设置成精确的宽度,一旦用户给子 View 设置了
MATCH_PARENT(匹配父窗体),子 View 的宽度应该是父窗体的宽度,而问题是子 View 并不知道父窗体多宽。 2.为了解决这样的矛盾,Android 系统干脆规定:子 View 的宽高必须交给父 View 去测量。在测量时,父 View 会根据自己的尺寸和子 View 的LayoutParams,计算出一个MeasureSpec(测量规则),也是父 View 对子 View 尺寸的期望。
3.然后遍历调用子 View 的measure(widthMeasureSpec,heightMeasureHeight)进行实际测量。测量完成,子 View 的宽高就确定了,可以通过view.getMeasuredWidth()方法获取。整个过程如下图所示。
2.能简单说下测量的过程吗
3.我们都知道测量中有三种模式 EXACTLY AT_MOST UNSPECIFIED 说下他们的意义
AT_MOST:最大模式
用这个模式去测量,并在测量时提供一个最大值,那么 View 就先以自己的内容为准,在不超过最大值的前提下,最终确定自己的尺寸
代码
int widthSpec = MeasureSpec.makeMeasureSpec(100, MeasureSpec.AT_MOST);
// 第二个参数为高度的测量规则,写 0 表示省略
view.measure(widthSpec, 0);
UNSPECIFIED:不确定模式
用这个模式去测量, View 就会任意确定自己的尺寸,不管你传什么值进去都是没有意义。很少用,为了避免和
AT_MOST搞混,暂时将它忽略。
int widthSpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
// 第二个参数为高度的测量规则,写 0 表示省略
view.measure(widthSpec, 0);
4.测量规则是什么呢 widthMeasureSpec,heightMeasureSpec 我们知道他们是32位,32位中代表的含义
widthMeasureSpec和heightMeasureSpec是测量规则- 是一个
int类型,但是它并不是实际的尺寸,而是尺寸和测量模式的合成值。- 它在
int类型的 32 位二进制位中,31-30 这两位表示模式,0~29 这三十位表示宽和高的实际值。- 通过
MeasureSpec类提供的静态方法,我们可以从widthMeasureSpec和heightMeasureSpec中提取测量模式和期望尺寸。
int mode = MeasureSpec.getMode(widthMeasureSpec);
int size = MeasureSpec.getSize(widthMeasureSpec);
5.如何确定的子类的测量模式
上面说了,父类测量子类是调用子 View 的
measure(widthMeasureSpec,heightMeasureHeight),这个两个测量widthMeasureSpec规则是父类根据自己的宽度和子类的LayoutParams计算出来的。 那么到底怎么计算测量规则,看下面一张图就全明白了(注意相同的颜色)。
推荐
Android 核心笔记笔录:https://github.com/733gh/GH-Android-Review-master