View体系(七)理解 MeasureSpec

MeasureSpecView的内部类,MeasureSpec 封装了从父级传递到子级的测量要求。每个 MeasureSpec 代表对宽度或高度的要求。 MeasureSpec 由大小和模式两部分组成。

更详细的说明请看 Google官方文档

下面是MeasureSpec的源码:(基于Android 12

    public static class MeasureSpec {
        private static final int MODE_SHIFT = 30;
        private static final int MODE_MASK  = 0x3 << MODE_SHIFT;

        public static final int UNSPECIFIED = 0 << MODE_SHIFT;

        public static final int EXACTLY     = 1 << MODE_SHIFT;

        public static final int AT_MOST     = 2 << MODE_SHIFT;

        public static int makeMeasureSpec(@IntRange(from = 0, to = (1 << MeasureSpec.MODE_SHIFT) - 1) int size,
                                          @MeasureSpecMode int mode) {
            if (sUseBrokenMakeMeasureSpec) {
                return size + mode;
            } else {
                return (size & ~MODE_MASK) | (mode & MODE_MASK);
            }
        }

        public static int makeSafeMeasureSpec(int size, int mode) {
            if (sUseZeroUnspecifiedMeasureSpec && mode == UNSPECIFIED) {
                return 0;
            }
            return makeMeasureSpec(size, mode);
        }

        public static int getMode(int measureSpec) {
            return (measureSpec & MODE_MASK);
        }

        public static int getSize(int measureSpec) {
            return (measureSpec & ~MODE_MASK);
        }

        static int adjust(int measureSpec, int delta) {
            final int mode = getMode(measureSpec);
            int size = getSize(measureSpec);
            if (mode == UNSPECIFIED) {
                return makeMeasureSpec(size, UNSPECIFIED);
            }
            size += delta;
            if (size < 0) {
                Log.e(VIEW_LOG_TAG, "MeasureSpec.adjust: new size would be negative! (" + size +
                        ") spec: " + toString(measureSpec) + " delta: " + delta);
                size = 0;
            }
            return makeMeasureSpec(size, mode);
        }

        public static String toString(int measureSpec) {
            int mode = getMode(measureSpec);
            int size = getSize(measureSpec);

            StringBuilder sb = new StringBuilder("MeasureSpec: ");

            if (mode == UNSPECIFIED)
                sb.append("UNSPECIFIED ");
            else if (mode == EXACTLY)
                sb.append("EXACTLY ");
            else if (mode == AT_MOST)
                sb.append("AT_MOST ");
            else
                sb.append(mode).append(" ");

            sb.append(size);
            return sb.toString();
        }
    }

从源码可以看出,它代表了32位int值,其中高2位代表mode,低30位代表sizemode指测量模式,size指测量值。mode有3种模式:

  • UNSPECIFIED:未指定模式,父View没有对子View施加任何限制。它可以是任何它想要的大小。

  • EXACTLY:精确模式,父View已经确定了子View的确切尺寸。无论子View想要多大,都将获得这些界限。

  • AT_MOST:最大模式,子View可以根据需要达到指定的最大的大小。

makeMeasureSpec用来保存宽和高的信息,getMode用来获取测量模式,getSize用来获取测量大小。

MeasureSpec受自身LayoutParams和父ViewMeasureSpec共同影响。作为顶层ViewDecorView来说,其并没有父View,那它的MeasureSpec是如何得来的呢?我们回到ViewRootImplperformTraversals方法:


                if (!mStopped || wasReportNextDraw) {
                boolean focusChangedDueToTouchMode = ensureTouchModeLocally(
                        (relayoutResult&WindowManagerGlobal.RELAYOUT_RES_IN_TOUCH_MODE) != 0);
                if (focusChangedDueToTouchMode || mWidth != host.getMeasuredWidth()
                        || mHeight != host.getMeasuredHeight() || dispatchApplyInsets ||
                        updatedConfiguration) {
                    int childWidthMeasureSpec = getRootMeasureSpec(mWidth, lp.width);//1
                    int childHeightMeasureSpec = getRootMeasureSpec(mHeight, lp.height);

                    if (DEBUG_LAYOUT) Log.v(mTag, "Ooops, something changed!  mWidth="
                            + mWidth + " measuredWidth=" + host.getMeasuredWidth()
                            + " mHeight=" + mHeight
                            + " measuredHeight=" + host.getMeasuredHeight()
                            + " dispatchApplyInsets=" + dispatchApplyInsets);

                    performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);//2

再看注释1处的getRootMeasureSpec方法:

    private static int getRootMeasureSpec(int windowSize, int rootDimension) {
        int measureSpec;
        switch (rootDimension) {

        case ViewGroup.LayoutParams.MATCH_PARENT:
            measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.EXACTLY);
            break;
        case ViewGroup.LayoutParams.WRAP_CONTENT:
            measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.AT_MOST);
            break;
        default:
            measureSpec = MeasureSpec.makeMeasureSpec(rootDimension, MeasureSpec.EXACTLY);
            break;
        }
        return measureSpec;
    }

第一个参数windowSize指窗口的尺寸,所以对于DecorView来说,它的MeasureSpec由自身的LayoutParams和窗口的尺寸决定,这一点和普通的View是不同的。
再回到上面看注释2处的performMeasure方法:

    private void performMeasure(int childWidthMeasureSpec, int childHeightMeasureSpec) {
        if (mView == null) {
            return;
        }
        Trace.traceBegin(Trace.TRACE_TAG_VIEW, "measure");
        try {
            mView.measure(childWidthMeasureSpec, childHeightMeasureSpec);//1
        } finally {
            Trace.traceEnd(Trace.TRACE_TAG_VIEW);
        }
    }

注释1处调用了mView.measure方法,进入了viewmeasure流程。

学习更多知识,请关注我的个人博客:droidYu

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
如果 `MeasureSpec.getSize()` 方法返回的结果为 0,可能是由于以下几种情况导致的: 1. 测量模式不正确:`MeasureSpec` 是用于测量视图尺寸的类,它包含了尺寸的测量模式和测量值。在调用 `MeasureSpec.getSize()` 方法之前,需要确保正确设置了测量模式。常见的测量模式有三种:`UNSPECIFIED`、`EXACTLY` 和 `AT_MOST`。如果测量模式为 `UNSPECIFIED`,则获取到的尺寸可能为 0。确保在测量过程中设置适当的测量模式。 2. 测量时机不正确:在获取尺寸之前,需要确保视图已经经过测量过程。在自定义 View 的 `onMeasure()` 方法中进行测量,并使用 `setMeasuredDimension()` 方法设置测量结果。如果在测量之前或者还没有完成测量过程就调用了 `MeasureSpec.getSize()` 方法,可能会得到 0 的结果。确保在合适的时机获取尺寸。 3. 视图尺寸未确定:如果视图的尺寸是根据内容动态确定的,并且在获取尺寸的时刻还没有完成布局过程,那么获取到的尺寸可能为 0。可以考虑在布局完成后再获取尺寸,例如使用 `ViewTreeObserver` 的监听器来监听布局完成事件,并在事件回调中获取尺寸。 4. 布局属性不正确:如果自定义 View 的布局属性设置不正确,可能会导致测量时无法获得准确的尺寸。确保自定义 View 在布局文件中正确设置了宽度和高度属性,并且在父布局中使用适当的布局参数。 如果以上方法仍然无法解决问题,可以进一步检查自定义 View 的代码,确保没有其他因素导致获取尺寸为 0。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值