從原始碼的角度分析,getWidth() 與 getMeasuredWidth() 的不同之處

亦楓發表於2017-04-07

之前不是寫了篇名為 Android 獲取 View 寬高的常用正確方式,避免為零 的總結性文章嘛,在結尾簡單闡述 measuredWidth 與 width 的區別。考慮到文章的重點,簡單幾筆帶過。沒曾想,引發一些爭論,大家對 View 的這兩對寬高屬性理解各有異議。於是便想追根溯源,通過解讀原始碼的方式說明一下,消除許多人的誤解。

備註:由於 height 和 measuedHeight 原理也都一樣,為了精簡語言,就不再重複敘說。

width 和 measuredWidth 的誤解


先來看看大家誤解的點在哪裡。很多人包括網上很多資料也都是這麼介紹的,初學 Android 時我也曾被這樣的言論誤導過:

width 表示 View 在螢幕上可顯示的區域大小,measuredWidth 表示 View 的實際大小,包括超出螢幕範圍外的尺寸;甚至有這樣的公式總結到:

getMeasuredWidth() = visible width + invisible width;

一個簡單的例子就足以說明這樣的解釋是錯誤的。寫一個簡單的佈局檔案:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <TextView
        android:id="@+id/tv_sample"
        android:layout_width="tv_sample"
        android:layout_height="wrap_content"
        android:text="This is a sample."
        android:background="@android:color/darker_gray"/>

</LinearLayout>複製程式碼

包含一個寬高自適應的 TextView 控制元件,在 Activity 中通過下面的程式碼獲取 width 和 measuredWidth 屬性值:

@Override
public void onWindowFocusChanged(boolean hasFocus) {
    super.onWindowFocusChanged(hasFocus);
    Log.i("size", "The width is " + mSampleTv.getWidth());
    Log.i("size", "The measured width is " + mSampleTv.getMeasuredWidth());
}複製程式碼

執行結果如圖:

從原始碼的角度分析,getWidth() 與 getMeasuredWidth() 的不同之處

logcat 控制檯列印如下:

04-04 16:25:52.191 31669-31669/com.yifeng.samples I/size: The width is 314
04-04 16:25:52.191 31669-31669/com.yifeng.samples I/size: The measured width is 314複製程式碼

這裡所用裝置的螢幕尺寸為 1080x1920,足以容納這個 TextView 的寬度。Log 顯示,width 和 measuredWidth 大小相同,均為 314 px。然後修改 android:layout_width 屬性值,使其超出螢幕寬度,同時,將文字內容加長一些:

...
<TextView
    android:id="@+id/tv_sample"
    android:layout_width="2000px"
    android:layout_height="wrap_content"
    android:text="This is a long sample.This is a long sample.This is a long sample.This is a long sample."
    android:background="@android:color/darker_gray"/>
...複製程式碼

再次執行,效果如圖:

從原始碼的角度分析,getWidth() 與 getMeasuredWidth() 的不同之處

顯然,文字內容已經超過螢幕寬度,顯示到螢幕之外的區域。如果按照前面的言論的話,width 應該為螢幕上顯示的寬度,即 1080 px,而 measuredWidth 肯定大於 width。事實真的如此嗎,請看 log 日誌:

04-04 16:36:47.329 6974-6974/com.yifeng.samples I/size: The width is 2000
04-04 16:36:47.330 6974-6974/com.yifeng.samples I/size: The measured width is 2000複製程式碼

width 等於 measuredWidth,都為 2000 px,事與願違,與我們想象的不一樣,前面的言論也就不攻自破。那到底 width 和 measuredWidth 有什麼區別呢,我們從原始碼的角度跟進一下。

getMeasuredWidth 原始碼剖析


先看 getMeasuredWidth() 方法的原始碼:

public final int getMeasuredWidth() {
    return mMeasuredWidth & MEASURED_SIZE_MASK;
}複製程式碼

這裡有個與運算,其中,MEASURED_SIZE_MASK 是個常量值:

public static final int MEASURED_SIZE_MASK = 0x00ffffff;複製程式碼

換算成二進位制是:111111111111111111111111,在與運算中,1 與任何數字進行與運算的結果都取決於對方。所以,mMeasuredWidth 變數值決定了 getMeasuredWidth() 方法的返回值。進一步檢視,mMeasuredWidth 的賦值在這裡:

private void setMeasuredDimensionRaw(int measuredWidth, int measuredHeight) {
    mMeasuredWidth = measuredWidth;
    mMeasuredHeight = measuredHeight;

    mPrivateFlags |= PFLAG_MEASURED_DIMENSION_SET;
}複製程式碼

但這個方法是私有方法,在 View 類內部呼叫,在這裡:

protected final void setMeasuredDimension(int measuredWidth, int measuredHeight) {
    boolean optical = isLayoutModeOptical(this);
    // 只展示核心程式碼
    ...
    setMeasuredDimensionRaw(measuredWidth, measuredHeight);
}複製程式碼

可以看出,mMeasuredWidth 的賦值,即 getMeasuredWidth() 的取值最終來源於 setMeasuredDimension() 方法呼叫時傳遞的引數!在自定義 View 時測量並設定 View 寬高時經常用到。通常在 onMeasure() 方法中設定,可以翻看一下系統中的 TextView、LinearLayout 等方法,都是如此。

getWidth 原始碼剖析


再看 getWidth() 方法的原始碼:

public final int getWidth() {
    return mRight - mLeft;
}複製程式碼

mRight、mLeft 變數分別表示 View 相對父容器的左右邊緣位置,並且二者的賦值是通過 setFrame() 方法中的引數獲取的:

protected boolean setFrame(int left, int top, int right, int bottom) {
        ...
        mLeft = left;
        mTop = top;
        mRight = right;
        mBottom = bottom;
        ...
}複製程式碼

setFrame() 方法中有這麼一句註釋,表明該方法的呼叫來自 layout() 方法:

Assign a size and position to this view.

This is called from layout.

那麼我們再看一下 layout() 方法:

public void layout(int l, int t, int r, int b) {
   ... 
   boolean changed = isLayoutModeOptical(mParent) ?
            setOpticalFrame(l, t, r, b) : setFrame(l, t, r, b);
   ...
}複製程式碼

其中也確實呼叫 setFrame() 方法。所以,getWidth() 的取值最終來源於 layout() 方法的呼叫。通常,layout() 方法在 parent 中被呼叫,來確定 child views 在父容器中的位置,一般在自定義 ViewGroup 的 onLayout() 方法中呼叫。

使用場景小結


分析完原始碼,至少能夠知道:measuredWidth 值在 View 的 measure 階段決定的,是通過 setMeasuredDimension() 方法賦值的;width 值在 layout 階段決定的,是由 layout() 方法決定的。有一點需要注意,通常來講,View 的 width 和 height 是由 View 本身和 parent 容器共同決定的。

一般情況下,getWidth() 與 getMeasuredWidth() 的返回值是相同的。在自定義 ViewGroup 時,會在 onLayout() 方法中通過 child.getMeasuredWidth() 方法獲取 child views 的原始大小來設定其顯示區域(諸如 LinearLayout 之類的系統中的 ViewGroup 都是這麼做的);除此之外,我們都可以通過 getWidth() 方法獲取 View 的實際顯示寬度。

關於我:亦楓,部落格地址:yifeng.studio/,新浪微博:IT亦楓

微信掃描二維碼,歡迎關注我的個人公眾號:安卓筆記俠

不僅分享我的原創技術文章,還有程式設計師的職場遐想

從原始碼的角度分析,getWidth() 與 getMeasuredWidth() 的不同之處

相關文章