附: 测量onMeasure()说明

2019-02-19  本文已影响0人  努力生活的西鱼

onMeasure()过程

最外层的View没有被调用measure,里面的子View:ProcessBar和ScrollView等等没有被调用measure方法,没有测量就不知道大小,没有大小就没有宽和高,就不知道怎么画出来。

1. 测量过程

父亲测量子View,如果父亲是ViewGroup,父亲有义务测量每一个孩子。形成树状结构测量。如果是View测量自身,也就是设置自己的大小。

2. 问题:刚才其他显示图片的页面,我们没有测量大小为什么能显示?

当我们没有写measure方法的时候,第二个页面其实也是显示出来的;也就是说我们并没有去测量,我们只是在onLayout方法中强制设置每个MyViewPagerView的一级子View的固定大小,并没有考虑它自身要多大,它想要多大。

但是对于一级子View的子View我们是没有测量的,所以不能显示。

系统的ViewGroup(线性布局,相对布局),给子View指定大小的时候,会考虑子View想要多大,测量出来了,有多大才给多大。

public final int getWidth() {
    return mRight - mLeft;
}
public final int getHeight() {
    return mBottom - mTop;
}

宽度:右边的距离减掉左边的距离
高度:底部的距离减掉顶部的距离

3. 问题:先有子View的大小还是先有父View的大小?

不确定。
例如:一个按钮,外边是个线性布局,高和宽是包裹类型;现在是先知道子View才知道父View有多大,如果,父View是填充窗体,子View也是填充窗体,现在是先知道父View大小,才知道子View大小。

4. 所有的View的大小,仅计算一次是不能完成的,一般要计算多次。
protected void onMeasure(int widthMeasureSpec,int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec,heightMeasureSpec);
    System.out.println(widthMeasureSpec + ": " + heightMeasureSpec);
}

结果:

1073742144   1073742193

这两个数是很大的数,这个数字包含了宽度的信息,还包含了给孩子的包裹计算,还有填充计算等信息。这两个信息是可以分开来的。Int类型最大值是20多亿,表示屏幕宽度已经足够。1万个像素足够。

int类型在内存中占32位。

前两位表示宽度,后面的30位表示大小
int size = MeasureSpec.getSize(widthMeasureSpec);
int mode = MeasureSpec.getMode(widthMeasureSpec);
System.out.println(size+": " + mode);
320 : 1073741824

第一个是宽,第二个是一种模式

知识拓展:

它常用的三个函数:

  1. static int getMode(int measureSpec): 根据提供的测量值(格式)提取模式。
  2. static int getSize(int measureSpec): 根据提供的测量值提取大小值。
  3. static int makeMeasureSpec(int size,int mode): 根据提供的大小值和模式创建一个测量值(格式)。通常是在View组件的onMeasure方法里面调用。
onMeasure总结

系统的onMeasure中所干的事:

  1. 根据widthMeasureSpec求得宽度width,和父view给的模式
  2. 根据自身的宽度width和自身的padding值,相减,求得子View可以拥有的宽度newWidth
  3. 根据newWidth和模式求得一个新的MeasureSpec值。
  4. 用新的MeasureSpec来计算子View
上一篇下一篇

猜你喜欢

热点阅读