MeasureSpec类介绍及使用详解

我们知道View在屏幕上显示出来要先经过measure和layout. 在调用onMeasure(int widthSpec, int heightSpec)方法时,要涉及到MeasureSpec的使用,MeasureSpec有3种模式分别是UNSPECIFIED, EXACTLY和AT_MOST, 那么这些模式和我们平时设置的layout参数fill_parent, wrap_content有什么关系呢。经过代码测试就知道,当我们设置width或height为fill_parent时,容器在布局时调用子 view的measure方法传入的模式是EXACTLY,因为子view会占据剩余容器的空间,所以它大小是确定的。而当设置为 wrap_content时,容器传进去的是AT_MOST, 表示子view的大小最多是多少,这样子view会根据这个上限来设置自己的尺寸。当子view的大小设置为精确值时,容器传入的是EXACTLY, 而MeasureSpec的UNSPECIFIED模式目前还没有发现在什么情况下使用。

MeasureSpec它常用的三个函数:

 

  1.static int getMode(int measureSpec):根据提供的测量值(格式)提取模式(上述三个模式之一)

 

  2.static int getSize(int measureSpec):根据提供的测量值(格式)提取大小值(这个大小也就是我们通常所说的大小)

 

  3.static int makeMeasureSpec(int size,int mode):根据提供的大小值和模式创建一个测量值(格式)

 

  这个类的使用呢,通常在view组件的onMeasure方法里面调用但也有少数例外,看看几个例子:

    a.首先一个我们常用到的一个有用的函数,View.resolveSize(int size,int measureSpec)

 public static int resolveSize(int size, int measureSpec) 
 {
   
         int result = size;         
   
                 int specMode = MeasureSpec.getMode(measureSpec);        
   
                 int specSize =  MeasureSpec.getSize(measureSpec);  
   
                  
                 switch (specMode) 
                 {        
   
                         case MeasureSpec.UNSPECIFIED:             
                                 result = size;           
   
                                 break;
   
                                  
                         case MeasureSpec.AT_MOST:             
                                 result = Math.min(size, specSize);            
   
                                 break;
   
                                  
                         case MeasureSpec.EXACTLY:             
                                 result = specSize;            
   
                                 break;         
   
                 }
   
                           
                 return result;
   
 }

上面既然要用到measureSpec值,那自然表示这个函数通常是在onMeasure方法里面调用的。简单说一下,这个方法的主要作用就是根据你提供的大小和模式,返回你想要的大小值,这个里面根据传入模式的不同来做相应的处理。

  再看看MeasureSpec.makeMeasureSpec方法,实际上这个方法很简单

public static int makeMeasureSpec(int size, int mode) 
{            
 
        return size + mode;         
 
}

这样大家不难理解size跟measureSpec区别了。看看它的使用吧,ListView.measureItem(View child)

 private void measureItem(View child) 
 {
   
          ViewGroup.LayoutParams p = child.getLayoutParams();
   
          if (p == null) 
                  {
   
               p = new ViewGroup.LayoutParams(
   
                       ViewGroup.LayoutParams.MATCH_PARENT,
   
                       ViewGroup.LayoutParams.WRAP_CONTENT);
   
          }
   
                   
   
          int childWidthSpec = ViewGroup.getChildMeasureSpec(mWidthMeasureSpec,
   
                  mListPadding.left + mListPadding.right, p.width); 
              int lpHeight = p.height;
   
                  int childHeightSpec;
   
                   
   
         if (lpHeight > 0) 
                 {
   
               childHeightSpec = MeasureSpec.makeMeasureSpec(lpHeight, MeasureSpec.EXACTLY);
   
         } 
           else
           {
   
                  childHeightSpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
   
            }
   
       child.measure(childWidthSpec, childHeightSpec);
   
 }

measureSpec方法通常在ViewGroup中用到,它可以根据模式(MeasureSpec里面的三个)可以调节子元素的大小。

  注意,使用EXACTLY和AT_MOST通常是一样的效 果,如果你要区别他们,那么你就要使用上面的函数View.resolveSize(int size,int measureSpec)返回一个size值,然后使用你的view调用setMeasuredDimension(int,int)函数。

 protected final void setMeasuredDimension(int measuredWidth, int measuredHeight)
   
 {
   
          mMeasuredWidth = measuredWidth;
   
          mMeasuredHeight = measuredHeight;
   
    
   
          mPrivateFlags |= MEASURED_DIMENSION_SET;     
   
 }

然后你调用view.getMeasuredWidth,view.getMeasuredHeigth 返回的就是上面函数里的mMeasuredWidth,mMeasuredHeight的值。