標籤:android style blog color 使用 os
註:根據網上資料整理如下
首先 onMeasure方法是為了得到各個View大小的函數
fill_parent-->public static final int EXACTLY = 1 << MODE_SHIFT;
wrap_content-->public static final int AT_MOST = 2 << MODE_SHIFT;
這是makeMeasureSpec方法的程式碼片段
public static int makeMeasureSpec(int size, int mode) { if (sUseBrokenMakeMeasureSpec) { return size + mode; } else { return (size & ~MODE_MASK) | (mode & MODE_MASK); } }
其中sUseBrokenMakeMeasureSpec的預設值是false:
/** * Use the old (broken) way of building MeasureSpecs. */ private static boolean sUseBrokenMakeMeasureSpec = false;
下面看一下sUseBrokenMakeMeasureSpec相關代碼
1 public View(Context context) { 2 //此處省略無關代碼......17 18 if (!sCompatibilityDone && context != null) {19 final int targetSdkVersion = context.getApplicationInfo().targetSdkVersion;20 21 // Older apps may need this compatibility hack for measurement.22 sUseBrokenMakeMeasureSpec = targetSdkVersion <= JELLY_BEAN_MR1;23 24 // Older apps expect onMeasure() to always be called on a layout pass, regardless25 // of whether a layout was requested on that View.26 sIgnoreMeasureCache = targetSdkVersion < KITKAT;27 28 sCompatibilityDone = true;29 }30 }
在構造方法中對sUseBrokenMakeMeasureSpec的值進行了判斷,代碼走進第18行又出現了一個sCompatibilityDone,
追溯其值原來預設值是false,所以在view初始化的時候肯定會走進這一行的,當sdk版本小於JELLY_BEAN_MR1也就是4.2的時候會返回true,
好 現在回頭看看也就是說4.2版本或一下makeMeasureSpec傳回值是size + mode之後的版本都是(size & ~MODE_MASK) | (mode & MODE_MASK)。
現在說一下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; (注意:二進位的加法,不是10進位的加法!) // 這裡設計的目的就是使用一個32位的位元,32和31位代表了mode的值,後30位代表size的值 // 例如size=100(4),mode=AT_MOST,則measureSpec=100+10000...00=10000..00100
//注意:最新版的sdk的makeMeasureSpec方法在文章首處,不過看過前文相信理解不難 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); }