关于android:理解Android中的MeasureSpec

44次阅读

共计 5544 个字符,预计需要花费 14 分钟才能阅读完成。

PS:本文系转载文章,浏览原文可读性会更好,文章开端有原文链接

ps:本文源码是基于 Android Api 31 来剖析的

目录

1、MeasureSpec

 1、1 SpecMode

 1、2 MeasureSpec 的 int 值和 LayoutParams 的对应关系


1、MeasureSpec

咱们在 Android 手机上看到的界面,其实就是 View,显示 View 的过程中其实是要通过 View 的测量的,View 的测量就须要用到 MeasureSpec,那么这个 MeasureSpec 是什么呢?MeasureSpec 翻译成中文就是“测量规格”或者“测量说明书”,MeasureSpec 在肯定水平上决定了一个 View 的尺寸规格,为什么说是肯定水平上?是因为这个过程还受父容器的影响,父容器影响 View 的 MeasureSpec 的创立过程;在测量过程中,零碎会将 View 的 LayoutParams 依据父容器所加限度的规定转换成对应的 MeasureSpec,最初依据这个 MeasureSpec 来测量出 View 的测量宽高。

这里的 MeasureSpec 有 2 层意思,一种是 MeasureSpec 对象,一种是 MeasureSpec 的 int 值;咱们说一下 MeasureSpec 的 int 值,MeasureSpec 的 int 值是 32 位的对不对?那好,高 2 位就是 SpecMode,低 30 位就是 SpecSize;SpecMode 和 SpecSize 又是什么东东呢?SpecMode 是测量模式,SpecSize 是某种测量模式下的测量大小。

为了更好的了解 MeasureSpec 的 int 值,咱们先看一下 View 的动态外部类 MeasureSpec;

图片

看正文 1,makeMeasureSpec 办法中的参数 size 和 mode 其实就是 SpecSize 和 SpecMode,MeasureSpec 对象通过 makeMeasureSpec 办法将 SpecSize 和 SpecMode 打包成一个 MeasureSpec 的 int 值来防止过多的对象内存调配;看正文 2 和正文 3,MeasureSpec 对象能够通过解包的模式得出它原始的 SpecMode 和 SpecSize。

1、1 SpecMode

SpecMode 有三类,每一类都有它的含意,咱们先看一下 View 的外部类 MeasureSpec 对 SpecMode 申明的是哪三类;

图片

看到正文 4、5、6 没有,SpecMode 的三类别离是 UNSPECIFIED、EXACTLY 和 AT_MOST,上面对它们的含意进行阐明一下;

UNSPECIFIED:父容器不对子 View 做任何的限度,想要多大就有多大,个别用于零碎外部应用。

EXACTLY:父容器曾经测量出子 View 须要的准确大小,子 View 的最终大小其实就是 SpecSize 所指定的值,它对应于 match_parent 和具体的数值这两种模式;具体的数值应该了解吧?比如说 200dp 的数值。

AT_MOST:父容器指定了一个可用大小 SpecSize,子 View 的大小不能超过这个值(SpecSize),具体是什么值要看子 View 的子 View 的大小,它对应于 wrap_content 这个值。

1、2 MeasureSpec 的 int 值和 LayoutParams 的对应关系

这里对 MeasureSpec 的转换有 2 种,一种是顶级的 View(DecorView),一种是一般的 View,咱们先看顶级 View 的 MeasureSpec 的转换是怎么样的;先看 2 段代码,看一下 ViewRootImpl 的 measureHierarchy 办法和 getRootMeasureSpec 办法;

图片

图片

看正文 7 和正文 8,参数 desiredWindowWidth 和 desiredWind-owHeighgt 就是屏幕尺寸大小,参数 lp.width 和 lp.height 就是子 View 的 LayoutParams,所以顶级的 View(DecorView)的 MeasureSpec 是由窗口的大小和本身的 View(DecorView)来决定的。

看正文 9、10、11,DecorView 的 MeasureSpec 就由 ViewGroup.Layout-Params.MATCH_PARENT、ViewGroup.LayoutParams.WRAP_CONTENT 和固定大小(正文 11 的代码就是固定大小)来决定了;其中正文 9 中的 ViewGroup.LayoutParams.MATCH_PARENT 示意准确模式,大小就是窗口的大小;正文 10 中的 ViewGroup.LayoutParams.WRAP_CONTENT 示意最大模式,但窗口大小不定,不能超过窗口的大小;正文 11 中的固定大小,比方为 50dp,它是准确模式,大小为 LayoutParams 指定的大小。

一般的 View 的 MeasureSpec 是怎么转换的呢?咱们晓得子 View 的 measure 过程是由父元素 ViewGroup 传递过去的,而父元素 ViewGroup 传递是在 ViewGroup 的 measureChildWithMargins 办法;

图片

看正文 12 和正文 13,ViewGroup 通过 getChildMeasureSpec 办法来失去子 View 的 MeasureSpec;从入参来看,子 View 的 MeasureSpec 的转换不只与父元素的 MeasureSpec 和子 View 自身的 LayoutParams 无关,

还与 子 View 的 margin 及 padding 无关。

咱们再具体看一下 ViewGroup 的 getChildMeasureSpec 办法;

public static int getChildMeasureSpec(int spec, int padding, int childDimension) {

    ......
    //14、int size = Math.max(0, specSize - padding);
    ......
    switch (specMode) {
        
        //15、case View.MeasureSpec.EXACTLY:
            if (childDimension >= 0) {
                resultSize = childDimension;
                resultMode = View.MeasureSpec.EXACTLY;
            } else if (childDimension == ViewGroup.LayoutParams.MATCH_PARENT) {
                // Child wants to be our size. So be it.
                resultSize = size;
                resultMode = View.MeasureSpec.EXACTLY;
            } else if (childDimension == ViewGroup.LayoutParams.WRAP_CONTENT) {
                // Child wants to determine its own size. It can't be
                // bigger than us.
                resultSize = size;
                resultMode = View.MeasureSpec.AT_MOST;
            }
            break;

        //16、case View.MeasureSpec.AT_MOST:
            if (childDimension >= 0) {
                // Child wants a specific size... so be it
                resultSize = childDimension;
                resultMode = View.MeasureSpec.EXACTLY;
            } else if (childDimension == ViewGroup.LayoutParams.MATCH_PARENT) {
                // Child wants to be our size, but our size is not fixed.
                // Constrain child to not be bigger than us.
                resultSize = size;
                resultMode = View.MeasureSpec.AT_MOST;
            } else if (childDimension == ViewGroup.LayoutParams.WRAP_CONTENT) {
                // Child wants to determine its own size. It can't be
                // bigger than us.
                resultSize = size;
                resultMode = View.MeasureSpec.AT_MOST;
            }
            break;

        //17、case View.MeasureSpec.UNSPECIFIED:
            if (childDimension >= 0) {
                // Child wants a specific size... let them have it
                resultSize = childDimension;
                resultMode = View.MeasureSpec.EXACTLY;
            } else if (childDimension == ViewGroup.LayoutParams.MATCH_PARENT) {
                // Child wants to be our size... find out how big it should
                // be
                resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;
                resultMode = View.MeasureSpec.UNSPECIFIED;
            } else if (childDimension == ViewGroup.LayoutParams.WRAP_CONTENT) {
                // Child wants to determine its own size.... find out how
                // big it should be
                resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;
                resultMode = View.MeasureSpec.UNSPECIFIED;
            }
            break;
    }
    //noinspection ResourceType
    return View.MeasureSpec.makeMeasureSpec(resultSize, resultMode);

}

看正文 14,子 View 的可用大小为父元素的尺寸减去 padding。

看正文 15,parentMeasureSpec 为 View.MeasureSpec.EXACTLY 模式,当 childDimension >= 0 时 就是子 View 的固定准确值(假如 100px),那么子 View 的 SpecSize(值为 childDimension)就是 childSize,SpecMode 就是 View.MeasureSpec.EXACTLY;当 childDimension == ViewGroup.LayoutParams.MATCH_PARENT 时,子 View 的 SpecSize 就为父元素的大小,SpecMode 就是 View.MeasureSpec.EXACTLY;当 childDimension == ViewGroup.LayoutParams.WRAP_CONTENT 时,子 View 的 SpecSize 就是父元素的大小,SpecMode 就是 View.MeasureSpec.AT_MOST。

看正文 16,parentMeasureSpec 为 View.MeasureSpec.AT_MOST 模式,当 childDimension >= 0 时 就是子 View 的固定准确值(假如 100px),那么子 View 的 SpecSize(值为 childDimension)就是 childSize,SpecMode 就是 View.MeasureSpec.EXACTLY;当 childDimension == ViewGroup.LayoutParams.MATCH_PARENT 时,子 View 的 SpecSize 就为父元素的大小,SpecMode 就是 View.MeasureSpec.AT_MOST;当 childDimension == ViewGroup.LayoutParams.WRAP_CONTENT 时,子 View 的 SpecSize 就是父元素的大小,SpecMode 就是 View.MeasureSpec.AT_MOST。

看正文 17,parentMeasureSpec 为 View.MeasureSpec.UNSPECIFIED 模式,当 childDimension >= 0 时 就是子 View 的固定准确值(假如 100px),那么子 View 的 SpecSize(值为 childDimension)就是 childSize,SpecMode 就是 View.MeasureSpec.EXACTLY;当 childDimension == ViewGroup.LayoutParams.MATCH_PARENT 时,子 View 的 SpecSize 就为 0,SpecMode 就是 View.MeasureSpec.UNSPECIFIED;当 childDimension == ViewGro-up.LayoutParams.WRAP_CONTENT 时,子 View 的 SpecSize 就是 0,SpecMode 就是 View.MeasureSpec.UNSPECIFIED。

小结:当 View 采纳固定宽和高的时候,不论父容器的 MeasureSpec 是什么,View 的 MeasureSpec 都是准确模式并且其大小遵循 Layoutparams 中的大小。当 View 的宽和高是 match_parent 时,如果父容器的模式是精准模式,那么 View 也是精准模式并且其大小是父容器的残余空间;如果父容器是最大模式,那么 View 也是最大模式并且其大小不会超过父容器的残余空间。当 View 的宽和高是 wrap_content 时,不论父容器的模式是精准还是最大化,View 的模式总是最大化并且大小不能超过父容器的残余空间。

正文完
 0