Android-MeasureSpec那些事

原文:http://tryenough.com/android-MeasureSpec

Android系統控件無法滿足我們的需求,因此有必要自定義View。具體方法參見官方開發文檔:http://developer.android.com/guide/topics/ui/custom-components.html

MeasureSpec的簡介

MesureSpec可以理解爲測量View大小的依據。它由一個32位的int值組成,前兩位表示測量模式,後30位表示大小值

測量模式(Mode)的類型有3種:UNSPECIFIED、EXACTLY 和
AT_MOST。

原文:http://tryenough.com/android-MeasureSpec

Measure源碼分析

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;  
  
        // UNSPECIFIED的模式設置:0向左進位30 = 00後跟30個0,即00 00000000000
        // 通過高2位
        public static final int UNSPECIFIED = 0 << MODE_SHIFT;  
        
        // EXACTLY的模式設置:1向左進位30 = 01後跟30個0 ,即01 00000000000
        public static final int EXACTLY = 1 << MODE_SHIFT;  

        // AT_MOST的模式設置:2向左進位30 = 10後跟30個0,即10 00000000000
        public static final int AT_MOST = 2 << MODE_SHIFT;  
  
        /**
          * makeMeasureSpec()方法
          * 作用:根據提供的size和mode得到一個詳細的測量結果,即measureSpec
          **/ 
            public static int makeMeasureSpec(int size, int mode) {  
            
                return size + mode;  
            // measureSpec = size + mode;此爲二進制的加法 而不是十進制
            // 設計目的:使用一個32位的二進制數,其中:32和31位代表測量模式(mode)、後30位代表測量大小(size)
            // 例如size=100(4),mode=AT_MOST,則measureSpec=100+10000...00=10000..00100  

            }  
      
        /**
          * getMode()方法
          * 作用:通過measureSpec獲得測量模式(mode)
          **/    

            public static int getMode(int measureSpec) {  
             
                return (measureSpec & MODE_MASK);  
                // 即:測量模式(mode) = measureSpec & MODE_MASK;  
                // MODE_MASK = 運算遮罩 = 11 00000000000(11後跟30個0)
                //原理:保留measureSpec的高2位(即測量模式)、使用0替換後30位
                // 例如10 00..00100 & 11 00..00(11後跟30個0) = 10 00..00(AT_MOST),這樣就得到了mode的值

            }  
        /**
          * getSize方法
          * 作用:通過measureSpec獲得測量大小size
          **/       
            public static int getSize(int measureSpec) {  
             
                return (measureSpec & ~MODE_MASK);  
                // size = measureSpec & ~MODE_MASK;  
               // 原理類似上面,即 將MODE_MASK取反,也就是變成了00 111111(00後跟30個1),將32,31替換成0也就是去掉mode,保留後30位的size  
            } 

    }  

原文:http://tryenough.com/android-MeasureSpec

MeasureSpec值的計算

子view的大小(MeasureSpec值)由父view的MeasureSpec值 和 子view的LayoutParams屬性 共同決定,具體計算邏輯封裝在getChildMeasureSpec()裏.

/**
  * 源碼分析:getChildMeasureSpec()
  * 作用:根據父視圖的MeasureSpec & 佈局參數LayoutParams,計算單個子View的MeasureSpec
  * 注:子view的大小由父view的MeasureSpec值 和 子view的LayoutParams屬性 共同決定
  **/

    public static int getChildMeasureSpec(int spec, int padding, int childDimension) {  

         //參數說明
         * @param spec 父view的詳細測量值(MeasureSpec) 
         * @param padding view當前尺寸的的內邊距和外邊距(padding,margin) 
         * @param childDimension 子視圖的佈局參數(寬/高)

            //父view的測量模式
            int specMode = MeasureSpec.getMode(spec);     

            //父view的大小
            int specSize = MeasureSpec.getSize(spec);     
          
            //通過父view計算出的子view = 父大小-邊距(父要求的大小,但子view不一定用這個值)   
            int size = Math.max(0, specSize - padding);  
          
            //子view想要的實際大小和模式(需要計算)  
            int resultSize = 0;  
            int resultMode = 0;  
          
            //通過父view的MeasureSpec和子view的LayoutParams確定子view的大小  


            // 當父view的模式爲EXACITY時,父view強加給子view確切的值
           //一般是父view設置爲match_parent或者固定值的ViewGroup 
            switch (specMode) {  
            case MeasureSpec.EXACTLY:  
                // 當子view的LayoutParams>0,即有確切的值  
                if (childDimension >= 0) {  
                    //子view大小爲子自身所賦的值,模式大小爲EXACTLY  
                    resultSize = childDimension;  
                    resultMode = MeasureSpec.EXACTLY;  

                // 當子view的LayoutParams爲MATCH_PARENT時(-1)  
                } else if (childDimension == LayoutParams.MATCH_PARENT) {  
                    //子view大小爲父view大小,模式爲EXACTLY  
                    resultSize = size;  
                    resultMode = MeasureSpec.EXACTLY;  

                // 當子view的LayoutParams爲WRAP_CONTENT時(-2)      
                } else if (childDimension == LayoutParams.WRAP_CONTENT) {  
                    //子view決定自己的大小,但最大不能超過父view,模式爲AT_MOST  
                    resultSize = size;  
                    resultMode = MeasureSpec.AT_MOST;  
                }  
                break;  
          
            // 當父view的模式爲AT_MOST時,父view強加給子view一個最大的值。(一般是父view設置爲wrap_content)  
            case MeasureSpec.AT_MOST:  
                // 道理同上  
                if (childDimension >= 0) {  
                    resultSize = childDimension;  
                    resultMode = MeasureSpec.EXACTLY;  
                } else if (childDimension == LayoutParams.MATCH_PARENT) {  
                    resultSize = size;  
                    resultMode = MeasureSpec.AT_MOST;  
                } else if (childDimension == LayoutParams.WRAP_CONTENT) {  
                    resultSize = size;  
                    resultMode = MeasureSpec.AT_MOST;  
                }  
                break;  
          
            // 當父view的模式爲UNSPECIFIED時,父容器不對view有任何限制,要多大給多大
            // 多見於ListView、GridView  
            case MeasureSpec.UNSPECIFIED:  
                if (childDimension >= 0) {  
                    // 子view大小爲子自身所賦的值  
                    resultSize = childDimension;  
                    resultMode = MeasureSpec.EXACTLY;  
                } else if (childDimension == LayoutParams.MATCH_PARENT) {  
                    // 因爲父view爲UNSPECIFIED,所以MATCH_PARENT的話子類大小爲0  
                    resultSize = 0;  
                    resultMode = MeasureSpec.UNSPECIFIED;  
                } else if (childDimension == LayoutParams.WRAP_CONTENT) {  
                    // 因爲父view爲UNSPECIFIED,所以WRAP_CONTENT的話子類大小爲0  
                    resultSize = 0;  
                    resultMode = MeasureSpec.UNSPECIFIED;  
                }  
                break;  
            }  
            return MeasureSpec.makeMeasureSpec(resultSize, resultMode);  
        }  

總結:
當父view的模式爲UNSPECIFIED時(多見於ListView、GridView ),父容器不對view有任何限制,要多大給多大。此情況比較少見,這裏不展開討論,下面總結其餘兩種情況:

  • 1.子View指定大小值時:

Mode = MeasureSpec.EXACTLY;
Size = 指定的大小

  • 2.子View指定爲MATCH_PARENT時:

Mode = 父View此時的模式;
Size = 父View的大小 - padding

  • 3.子View指定爲WRAP_CONTENT時:

Mode = AT_MOST;
Size = 父View的大小 - padding, 即父View中剩餘的空間

原文:http://tryenough.com/android-MeasureSpec

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章