MeasureSpec

在自定义ViewViewGroup的时候,我们经常会遇到int型的MeasureSpec来表示一个组件的大小,这个变量里面不仅有组件的尺寸大小,还有大小的模式。

这个大小的模式,有点难以理解。在系统中组件的大小模式有三种:

1.精确模式(MeasureSpec.EXACTLY

在这种模式下,尺寸的值是多少,那么这个组件的长或宽就是多少。

2.最大模式(MeasureSpec.AT_MOST

这个也就是父组件,能够给出的最大的空间,当前组件的长或宽最大只能为这么大,当然也可以比这个小。

3.未指定模式(MeasureSpec.UNSPECIFIED这种情况不多,一般都是父控件是AdapterView,通过measure方法传入的模式。

这个就是说,当前组件,可以随便用空间,不受限制。

    可能有很多人想不通,一个int型整数怎么可以表示两个东西(大小模式和大小的值),一个int类型我们知道有32位。而模式有三种,要表示三种状  态,至少得2位二进制位。于是系统采用了最高的2位表示模式。如图:

最高两位是00的时候表示"未指定模式"。即MeasureSpec.UNSPECIFIED

最高两位是01的时候表示"'精确模式"。即MeasureSpec.EXACTLY

最高两位是11的时候表示"最大模式"。即MeasureSpec.AT_MOST

很多人一遇到位操作头就大了,为了操作简便,于是系统给我提供了一个MeasureSpec工具类。

这个工具类有四个方法和三个常量(上面所示)供我们使用:

 

//这个是由我们给出的尺寸大小和模式生成一个包含这两个信息的int变量,这里这个模式这个参数,传三个常量中的一个。

public static int makeMeasureSpec(int size, int mode)

 

//这个是得到这个变量中表示的模式信息,将得到的值与三个常量进行比较。

public static int getMode(int measureSpec)

 

//这个是得到这个变量中表示的尺寸大小的值。

public static int getSize(int measureSpec)

 

//把这个变量里面的模式和大小组成字符串返回来,方便打日志

 public static String toString(int measureSpec)



MeasureSpec.EXACTLY:使用measureSpec中size的值作为宽高的精确值

当我们将控件的layout_width或layout_height指定为具体数值时如andorid:layout_width="50dip",或者为FILL_PARENT是,都是控件大小已经确定的情况,都是精确尺寸。

        MeasureSpec.AT_MOST:使用measureSpec中size的值作为最大值,采用不超过这个值的最大允许值

        当控件的layout_width或layout_height指定为WRAP_CONTENT时,控件大小一般随着控件的子空间或内容进行变化,此时控件尺寸只要不超过父控件允许的最大尺寸即可。因此,此时的mode是AT_MOST,size给出了父控件允许的最大尺寸。

        MeasureSpec.UNSPECIFIED是未指定尺寸,这种情况不多



以scrollview嵌套listview为例,我们重写onMesure方法:

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    int width = MeasureSpec.makeMeasureSpec(1000>>2,MeasureSpec.AT_MOST);

    super.onMeasure(widthMeasureSpec, width);
}
1000的二进制:1111101000

右移2位后:11111010,十进制为:250

这样就指定了listview的高度为250px以内的最大允许值(一般就是250)

把AT_MOST改为EXACTLY,则精确指定listview高度值为250px,如果listview内容全部显示的高度为500px(大于250px),那么当measureSpec中size的值为250px(小于500px)时,效果是一样的

如果设置的measureSpec中size的值大于listview内容全部显示的高度,那么设置成AT_MOST时,最多显示listview内容全部显示的高度,而EXACTLY还是显示measureSpec中size的值,所以EXACTLY在这种情况下,后面会留有空白高度(measureSpec中size的值大于listview内容全部显示的高度的部分显示为空白

所以,一般这样写可以让listview正确测量:

int width = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE>>2,MeasureSpec.AT_MOST);
MAX_VALUE右移2位后,即使不是最大整数了,listview的高度也一般不可能超过它

第一个参数有个最大值的限制:1073741823(二进制的30个1),MAX_VALUE是1个0加上31个1(二进制),所以也可以右移1位,但是由于最前面两位表示mode,而不是size,所有右移1位和右移2位是一样的(前面两位的值都会被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
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值