今天我学习了MeasureSpec,写下这博客目的是让自己明白它的流程是怎样的,也帮助大家明白它的流程是怎样的。
什么是MeasureSpec?
从字面上的意思是:"测量规格",(它)MeasureSpec决定了View的一个尺寸大小,当然了决定View的尺寸大小还会受到父容器的影响,下面我用图片来展示。MeasureSpec代表一个32位int值(目的为了避免过多的对象内存分配),高2位代表SpecMode,低30位代表SpecSize。
SpecSize:指的是在某种测量模式下的规格大小
SpecMode: 指的是测量模式,一共有3种测量的模式
UNSPECIFIED:
父容器不对View有任何的限制,要多大给多大。
EXACTLY:
父容器已经检测出View所需要的精确大小,这个时候View的最终大小就是SpecSize所指定的值。它对应于LayoutParams中的match_parent和具体的数值这二种模式
AT_MOST:
父容器指定了一个可用的大小即SpecSize,View的大小不能大于这个值,它对应LayoutParams中的wrap_content
在看View的绘制方法之前(measure方法),我们先看,我们在这里先看下ViewGroup里面的measureChildWithMargins方法,因为是这个方法会去调用view里面的measure的方法的。
protected void measureChildWithMargins(View child,
int parentWidthMeasureSpec, int widthUsed,
int parentHeightMeasureSpec, int heightUsed) {
final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,
mPaddingLeft + mPaddingRight + lp.leftMargin + lp.rightMargin
+ widthUsed, lp.width);
final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec,
mPaddingTop + mPaddingBottom + lp.topMargin + lp.bottomMargin
+ heightUsed, lp.height);
//此处调用View的measure方法
child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
}
从上面我们看到我们获取了子元素的MeasureSpec 即(childWidthMeasureSpec/childHeightMeasureSpec )。从我们上面的代码来看,我们子元素的MeasureSpec,受到父元素的MeasureSpec(parentWidthMeasureSpec/parentHeightMeasureSpec)的影响和子元素本身的LayoutParams有关,还和view的 padding,和margin有关系。
这里面我们有必要研究下getChildMeasureSpec,因为它会决定childView的测量模式
getChildMeasureSpec:
spec:指的是parentMeasureSpec
padding:就是padding+margin
childDimension:view的layoutParams,也就是指定宽度/高度 比如30dp或者wrap、match
public static int getChildMeasureSpec(int spec, int padding, int childDimension) {
int specMode = MeasureSpec.getMode(spec);
int specSize = MeasureSpec.getSize(spec);//父容器的大小
int size = Math.max(0, specSize - padding);//获取剩下的空间
int resultSize = 0;
int resultMode = 0;
switch (specMode) {//父容器的测量模式
case MeasureSpec.EXACTLY://父容器是指定的精确大小,
if (childDimension >= 0) {//如果子容器也指定了精确大小,那么测量模式和
resultSize = childDimension;//父容器一样
resultMode = MeasureSpec.EXACTLY;
} else if (childDimension == LayoutParams.MATCH_PARENT) {//如果子容器是match,那么它的测量模式是精确大小,
resultSize = size;
resultMode = MeasureSpec.EXACTLY;
} else if (childDimension == LayoutParams.WRAP_CONTENT) {
resultSize = size;
resultMode = MeasureSpec.AT_MOST;
}
break;
case MeasureSpec.AT_MOST://父容器是指定一个大小,即specSize ,因为这里是父容器剩下的空间给子容器
if (childDimension >= 0) {//如果子容器有个确定的值
resultSize = childDimension;
resultMode = MeasureSpec.EXACTLY;
} else if (childDimension == LayoutParams.MATCH_PARENT) {//如果子容器是match
resultSize = size;
resultMode = MeasureSpec.AT_MOST;//这个就是wrap,范围不能超过specSize
} else if (childDimension == LayoutParams.WRAP_CONTENT) {//如果子容器是wrap
resultSize = size;
resultMode = MeasureSpec.AT_MOST;
}
break;
case MeasureSpec.UNSPECIFIED://这个就是针对是scrollView的时候采用的最多的时候
if (childDimension >= 0) {
resultSize = childDimension;
resultMode = MeasureSpec.EXACTLY;
} else if (childDimension == LayoutParams.MATCH_PARENT) {
resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;
resultMode = MeasureSpec.UNSPECIFIED;
} else if (childDimension == LayoutParams.WRAP_CONTENT) {
resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;
resultMode = MeasureSpec.UNSPECIFIED;
}
break;
}
return MeasureSpec.makeMeasureSpec(resultSize, resultMode);
}
总的来说,子容器的测量模式是收到父容器的测量模式影响,子容器的大小,受到父容器剩下空间(specSize)和子view的layoutParams印影响,最后我像大家展示一下表格的方式
此处parentSize指的是: int specSize = MeasureSpec.getSize(spec);//父容器的大小
childSize指的是:childDimension
我们可以通过源码知道子view采用固定宽高的时候,不管父容器的测量模式是哪种(MeasureSpec),子View都是都是精确模式(EXACTLY)并且遵循Layoutparams中的大小,如果父容器是精确模式(EXACTLY),子容器是Match,那么子容器就是精确模式(EXACTLY),且大小是父容器剩下的空间 int size = Math.max(0, specSize - padding);//获取剩下的空间
剩下的模式我就不为大家一一打字了,大家可以直接用这个来参考,来确定子View是采用的哪种测量模式
好,以上就是我所学到的MeasureSpec
学习资源书籍:《Android 开发艺术探索》