前言:前几天把onMeasure,onLayout的相关方法学习了一下,打算在做一个项目有深刻理解以后,在总结自己关于这两个方法的学习。学习当中,感觉又会遇到新的问题,新的要学习的知识点。感觉如果还不把之前学习的内容,记录下来,现在不抽出时间,那么以后也抽不出时间。所以,学习一个知识点,就要记录下,不要拖。或许,这就是解决拖延症的办法:做一件事就做彻底,不留尾巴。Just Do It.
首先明确一点,关于View的视图结构,是一个树形递归结构。不论我们是在执行onMeasure方法,还是onLayout方法,都是按照这个流程结构走的。无论我们测量、还是调整布局,都是从树的顶端,一层一层遍历,一个分支一个分支的执行measure,onMeasure,都测量完毕,在递归执行layout,onLayout。
所学onMeasure知识,关于对其总结,也是基于学习了各个博主的讲解,结合源码。最后给出相关博主连接。
Android 应用程序UI绘制分为三个过程(先后顺序是):测量measure,布局位置layout,绘制draw。
关于View的递归遍历,不仅是onMeasure适用,onLayout也一样适用。
一次遍历下来,第一个子控件以及这个子控件中的所有子控件都会完成测量工作;然后开始测量第二个子控件…;
最后父控件所有的子控件都完成测量以后会调用setMeasureDimension方法保存自己的测量大小。值得注意的是,
这个过程不只执行一次,也就是说有可能重复执行,因为有的时候,一轮测量下来,父控件发现某一个子控件的尺寸不符合要求,
就会重新测量一遍。上图:
一、在自定义View中,一般情况下,只有View的宽、高属性需要wrap_content的时候,才会用到onMeasure方法。
在View的源码中,measure(w,h)方法测量View的实际大小。是个final类型,不允许外部修改调用。所以继承View的扩展类,只能调用measure(w,h)里的onMeasure()方法,对布局进行测量。但是子View计算自己宽高时,可以调用该方法。
public final void measure(int widthMeasureSpec, int heightMeasureSpec) {
...
onMeasure(widthMeasureSpec, heightMeasureSpec);
...
}
看onMeasure()方法内部:
/**
* 这个方法需要被重写,应该由子类去决定测量的宽高值.onMeasure方法宽高得到的是父布局控件的宽高
*/
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),
getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));
}
在onMeasure()方法内部调用了setMeasuredDimension()方法。,作用是来存储测量宽,高值。作用是来存储测量宽,高值。作用是来存储测量宽,高值。(重要说三遍。)
/**
* 这个方法必须由onMeasure(int, int)来调用,来存储测量的宽,高值。
*/
protected final void setMeasuredDimension(int measuredWidth, int measuredHeight) {
mMeasuredWidth = measuredWidth;
mMeasuredHeight = measuredHeight;
mPrivateFlags |= PFLAG_MEASURED_DIMENSION_SET;
}
在返回onMeasure()方法内,getDefaultSize()方法。看内部方法:
/**
* 作用是返回一个默认的值,如果MeasureSpec没有强制限制的话则使用提供的大小.否则在允许范围内可任意指定大小
* 第一个参数size为提供的默认大小,第二个参数为测量的大小
*/
public static int getDefaultSize(int size, int measureSpec) {
int result = size;
int specMode = MeasureSpec.getMode(measureSpec);//取mode
int specSize = MeasureSpec.getSize(measureSpec);//取size
switch (specMode) {
// Mode = UNSPECIFIED时使用提供的默认大小
case MeasureSpec.UNSPECIFIED:
result = size;
break;
// Mode = AT_MOST,EXA