從源碼的角度分析,getWidth() 與 getMeasuredWidth() 的不一樣之處

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

備註:因爲 height 和 measuedHeight 原理也都同樣,爲了精簡語言,就再也不重複敘說。php

width 和 measuredWidth 的誤解


先來看看你們誤解的點在哪裏。不少人包括網上不少資料也都是這麼介紹的,初學 Android 時我也曾被這樣的言論誤導過:java

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

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 屬性值:ide

@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());
}複製代碼

運行結果如圖:佈局

logcat 控制檯打印以下:this

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 屬性值,使其超出屏幕寬度,同時,將文本內容加長一些:spa

...
<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"/>
...複製代碼

再次運行,效果如圖:

顯然,文本內容已經超過屏幕寬度,顯示到屏幕以外的區域。若是按照前面的言論的話,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亦楓

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

不只分享個人原創技術文章,還有程序員的職場遐想

相關文章
相關標籤/搜索