美文网首页Android知识
Android MeasureSpec 基础学习

Android MeasureSpec 基础学习

作者: 一个冬季 | 来源:发表于2017-11-12 12:04 被阅读75次

今天我学习了MeasureSpec,写下这博客目的是让自己明白它的流程是怎样的,也帮助大家明白它的流程是怎样的。

什么是MeasureSpec?
从字面上的意思是:"测量规格",(它)MeasureSpec决定了View的一个尺寸大小,当然了决定View的尺寸大小还会受到父容器的影响,下面我用图片来展示。 2.png

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

1(O[{JK~C`VS4GXGNP]7NLB.png
在看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印影响,最后我像大家展示一下表格的方式


3.png

此处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 开发艺术探索》

相关文章

网友评论

    本文标题:Android MeasureSpec 基础学习

    本文链接:https://www.haomeiwen.com/subject/jpcbmxtx.html