UI绘制流程之测量------onMeasure()方法之探讨实现不可滑动的原理

本文介绍如何通过重写onMeasure()方法实现ListView禁止滑动的效果。解析MeasureSpec类中makeMeasureSpec()方法的工作原理,并展示了具体的代码实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在自定义view的时候,只需要知道以下三个步骤:
1.测量——onMeasure():决定View的大小
2.布局——onLayout():决定View在ViewGroup中的位置
3.绘制——onDraw():如何绘制这个View。(非必须)

今天来总理下onMeasure()方法的理解:
首先onMeasure()方法到目前为止我最熟悉的是用途是用来让可以滑动的View不能滑动.
不知道你经常那他来做什么,现在就来看看一个很简单的不可滑动的ListView的写法,以此来浅谈一下onMeasure()的写法 :

public class NoScrollListview extends ListView{  

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

   /**
     * 测量宽高来设置不可滑动
     * @param widthMeasureSpec 宽的详细测量值
     * @param heightMeasureSpec 高的详细测量值 
     */
public void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
    {
    int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
    super.onMeasure(widthMeasureSpec, expandSpec);

}

可以看到onMeasure()方法接收两个参数宽高的详细测量值 ,至于什么叫详细测量值 , 这里不多解释 , 反正就是预计的宽高 ,可以理解为视图View想要的大小(宽高)说明(想要的未必就是最终大小)。

然后在方法内部调用了MeasureSpec类的makeMeasureSpec()方法 , 还好这个类不是很大 ,还好搜到一篇详细注释的这个类 (附录在最后), 现在一起来学习一下 这个方法:

  /** 
         * 根据提供的size和mode得到一个详细的测量结果 
         */  
        // measureSpec = size + mode;   (注意:二进制的加法,不是十进制的加法!)  
        // 这里设计的目的就是使用一个32位的二进制数,32和31位代表了mode的值,后30位代表size的值  
        // 例如size=100(4),mode=AT_MOST,则measureSpec=100+10000...00=10000..00100  
        public static int makeMeasureSpec(int size, int mode) {  
            return size + mode;  
        }  

这里详细介绍了makeMeausreSpec()方法 , 可以很清楚的明白我们改写ListView测量的最大高度为屏幕的高度,所以ListView就不能滑动了.

附录MeasureSpec类详解

/** 
 * MeasureSpec封装了父布局传递给子布局的布局要求,每个MeasureSpec代表了一组宽度和高度的要求 
 * MeasureSpec由size和mode组成。 
 * 三种Mode: 
 * 1.UNSPECIFIED 
 * 父不没有对子施加任何约束,子可以是任意大小(也就是未指定) 
 * (UNSPECIFIED在源码中的处理和EXACTLY一样。当View的宽高值设置为0的时候或者没有设置宽高时,模式为UNSPECIFIED 
 * 2.EXACTLY 
 * 父决定子的确切大小,子被限定在给定的边界里,忽略本身想要的大小。 
 * (当设置width或height为match_parent时,模式为EXACTLY,因为子view会占据剩余容器的空间,所以它大小是确定的) 
 * 3.AT_MOST 
 * 子最大可以达到的指定大小 
 * (当设置为wrap_content时,模式为AT_MOST, 表示子view的大小最多是多少,这样子view会根据这个上限来设置自己的尺寸) 
 *  
 * MeasureSpecs使用了二进制去减少对象的分配。 
 */  
public class MeasureSpec {  
        // 进位大小为2的30次方(int的大小为32位,所以进位30位就是要使用int的最高位和倒数第二位也就是32和31位做标志位)  
        private static final int MODE_SHIFT = 30;  

        // 运算遮罩,0x3为16进制,10进制为3,二进制为11。3向左进位30,就是11 00000000000(11后跟30个0)  
        // (遮罩的作用是用1标注需要的值,0标注不要的值。因为1与任何数做与运算都得任何数,0与任何数做与运算都得0)  
        private static final int MODE_MASK  = 0x3 << MODE_SHIFT;  

        // 0向左进位30,就是00 00000000000(00后跟30个0)  
        public static final int UNSPECIFIED = 0 << MODE_SHIFT;  
        // 1向左进位30,就是01 00000000000(01后跟30个0)  
        public static final int EXACTLY     = 1 << MODE_SHIFT;  
        // 2向左进位30,就是10 00000000000(10后跟30个0)  
        public static final int AT_MOST     = 2 << MODE_SHIFT;  

        /** 
         * 根据提供的size和mode得到一个详细的测量结果 
         */  
        // measureSpec = size + mode;   (注意:二进制的加法,不是十进制的加法!)  
        // 这里设计的目的就是使用一个32位的二进制数,32和31位代表了mode的值,后30位代表size的值  
        // 例如size=100(4),mode=AT_MOST,则measureSpec=100+10000...00=10000..00100  
        public static int makeMeasureSpec(int size, int mode) {  
            return size + mode;  
        }  

        /** 
         * 通过详细测量结果获得mode 
         */  
        // mode = measureSpec & MODE_MASK;  
        // MODE_MASK = 11 00000000000(11后跟30个0),原理是用MODE_MASK后30位的0替换掉measureSpec后30位中的1,再保留32和31位的mode值。  
        // 例如10 00..00100 & 11 00..00(11后跟30个0) = 10 00..00(AT_MOST),这样就得到了mode的值  
        public static int getMode(int measureSpec) {  
            return (measureSpec & MODE_MASK);  
        }  

        /** 
         * 通过详细测量结果获得size 
         */  
        // size = measureSpec & ~MODE_MASK;  
        // 原理同上,不过这次是将MODE_MASK取反,也就是变成了00 111111(00后跟30个1),将32,31替换成0也就是去掉mode,保留后30位的size  
        public static int getSize(int measureSpec) {  
            return (measureSpec & ~MODE_MASK);  
        }  

        /** 
         * 重写的toString方法,打印mode和size的信息,这里省略 
         */  
        public static String toString(int measureSpec) {  
            return null;  
        }  
}  
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值