美文网首页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