Android-MeasureSpec那些事
生活随笔
收集整理的這篇文章主要介紹了
Android-MeasureSpec那些事
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
原文:http://tryenough.com/android-...
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-...
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與任何數做與運算都得0private 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 00000000000public static final int EXACTLY = 1 << MODE_SHIFT; // AT_MOST的模式設置:2向左進位30 = 10后跟30個0,即10 00000000000public 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值的計算
子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-...
總結
以上是生活随笔為你收集整理的Android-MeasureSpec那些事的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 写在方法中的路由跳转
- 下一篇: 京瓷打印机几个常见密码