Android系統控件沒法知足咱們的需求,所以有必要自定義View。具體方法參見官方開發文檔:developer.android.com/guide/topic…html
MesureSpec能夠理解爲測量View大小的依據。它由一個32位的int值組成,前兩位表示測量模式,後30位表示大小值。java
測量模式(Mode)的類型有3種:UNSPECIFIED、EXACTLY 和 AT_MOST。android
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
}
}
複製代碼
子view的大小(MeasureSpec值)由父view的MeasureSpec值 和 子view的LayoutParams屬性 共同決定,具體計算邏輯封裝在getChildMeasureSpec()裏.ide
/** * 源碼分析: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有任何限制,要多大給多大。此狀況比較少見,這裏不展開討論,下面總結其他兩種狀況:源碼分析
Mode = MeasureSpec.EXACTLY; Size = 指定的大小佈局
Mode = 父View此時的模式; Size = 父View的大小 - paddingui
Mode = AT_MOST; Size = 父View的大小 - padding, 即父View中剩餘的空間spa