关于MeasureSpec的一些理解

MeasureSpec有以下三种模式:

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;

分别表示:

  1. 精确模式(MeasureSpec.EXACTLY) 
    在这种模式下,尺寸的值是多少,那么这个组件的长或宽就是多少。
  2. 最大模式(MeasureSpec.AT_MOST) 
    这个也就是父组件,能够给出的最大的空间,当前组件的长或宽最大只能为这么大,当然也可以比这个小。
  3. 未指定模式(MeasureSpec.UNSPECIFIED) 
    这个就是说,当前组件,可以随便用空间,不受限制。

 

我们在MeasureSpec这个类中可以看到:
private static final int MODE_SHIFT = 30;
private static final int MODE_MASK  = 0x3 << MODE_SHIFT;

其中MODE_MASK就是0x3(16进制的3)等同于10进制的3,转化为二进制就是 00000000 00000000 00000000 00000011 。右移30位变成二进制 11000000 00000000 00000000 00000000。

下面我们重点看看如下这个方法:

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);
            }
}
默认sUseBrokenMakeMeasureSpec = false;

我们时常重写ListView的方法如下:

public class MyListView extends ListView {

    public MyListView(Context context) {
        super(context);
    }

    public MyListView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public MyListView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec
                ,MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE>>2,MeasureSpec.AT_MOST));
    }

}

Integer.MAX_VALUE的二进制值为 11111111 11111111 11111111 11111111 ,>>2,二进制值变更为 00111111 11111111 11111111 11111111。

(size & ~MODE_MASK) | (mode & MODE_MASK);

~MODE_MASK 11000000 00000000 00000000 00000000

(size & ~MODE_MASK) 11111111 11111111 11111111 11111111

(mode & MODE_MASK) 11000000 00000000 00000000 00000000

所以返回值为 11111111 11111111 11111111 11111111  前面两位代表MODE。

  • 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
发出的红包

打赏作者

AD钙奶-lalala

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值