View工作原理 -- 工作过程 -- measure(3)

获取View的宽/高

在Activity的onCreate、onStart、onResume中均无法正确得到某个View的宽/高信息,这是因为View的measure过程和Activity的生命周期方法不是同步执行的,因此无法保证Activity执行了onCreate、onStart、onResume时某个View已经测量完毕了,如果View还没有测量完毕,那么获得的宽/高就是0。可以通过以下几种方法获取。

一、Activity/View#onWindowFocusChanged

onWindowFocusChanged方法的含义是:View已经初始化完毕,宽/高已经准备好了,这个时候去获取宽/高是没问题的。需要注意的是,当Activity的窗口得到焦点和失去焦点时均会被调用一次。

pubic void onWinowFocusChanged(boolean hasFocus) {
    super.onWinowFocusChanged(hasFocus);
    if(hasFocus) {
        int widht = view.getMeasuredWidth();
        int height = view.getMeasuredHeight();
    }
}

二、view.post(runnable)

通过post可以将runnable投递到消息队列的尾部,然后等待Looper调用此runnable的时候,View也已经初始化好了。

protected void onStart() {
    super.onStart();
    view.post(new Runnable() {
        @Override
        public void run() {
            int width = view.getMeasuredWidth();
            int height = view.getMeasureHeight();
        }
    });
}

三、ViewTreeObserver

使用ViewTreeObserver的众多回调可以完成这个功能,比如使用OnGlobalLayoutListener接口,当View树的状态发生改变或者View树内部的View的可见性发生改变时,onGlobalLayout方法将被回调。需要注意的是,伴随着View树的状态改变等,onGlobalLayout方法会被回调多次。

protected void onStart() {
    super.onStart();
    ViewTreeObserver observer = view.getViewTreeObserver();
    observer.addOnGlobalLayoutListener(new OnGlobalLayoutListener() {
        @SuppressWarnings("deprecation")
        @Override
        public void onGlobalLayout() {
            view.getViewTreeObserver().removeGlobalOnLayoutListener(this);
            int width = view.getMeasuredWidth();
            int height = view.getMeasuredHeight();
        }
    });
}

四、view.measure(int widthMeasureSpec, int heightMeasureSpec)

通过手动对View进行measure来得到View的宽/高。这种情况比较复杂,需要分情况处理,根据View的LayoutParams来分:
1.match_parent
无法measure出具体的宽/高。原因很简单,根据View的measure过程,构造此种MeasureSpec需要知道parentSize,即父容器的剩余空间,而这个时候我们无法知道parentSize,所以理论上不可能测量出View的大小。
2.固定大小
比如宽/高都是100px,代码如下:

int widthMeasureSpec = MeasureSpec.makeMeasureSpec(100, MeasureSpec.EXACTLY);
int heightMeasureSpec = MeasureSpec.makeMeasureSpec(100, MeasureSpec.EXACTLY);
view.measure(widthMeasureSpec, heightMeasureSpec);

3.wrap_content
代码如下:

int widthMeasureSpec = MeasureSpec.makeMeasureSpec((1 << 30) - 1, MeasureSpec.AT_MOST);
int heightMeasureSpec = MeasureSpec.makeMeasureSpec((1 << 30) - 1, MeasureSpec.AT_MOST);
view.measure(widthMeasureSpec, heightMeasureSpec);

在最大化模式下,用View理论上能支持的最大值去构造MeasureSpec是合理的。
例子:

//布局
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/main_layout"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingTop="100px">

    <TextView
        android:id="@+id/main_tv"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello, Android!"/>

    <Button
        android:id="@+id/main_btn"
        android:layout_width="200px"
        android:layout_height="wrap_content"
        android:layout_marginTop="100px"
        android:gravity="center"
        android:textAllCaps="false"
        android:text="Click"/>
</LinearLayout>

//代码
private void testMeasureSpec() {
    int widthMeasureSpec,heightMeasureSpec;
    Logger.d("main_btn原始测量尺寸, width: " + mMainBtn.getMeasuredWidth() + ", height: " + mMainBtn.getMeasuredHeight());
    //宽度设置SpecMode为EXACTLY,SpecSize为400
    widthMeasureSpec = View.MeasureSpec.makeMeasureSpec(400, View.MeasureSpec.EXACTLY);
    heightMeasureSpec = View.MeasureSpec.makeMeasureSpec((1 << 30) - 1, View.MeasureSpec.AT_MOST);
    mMainBtn.measure(widthMeasureSpec, heightMeasureSpec);
    Logger.d("宽度设置SpecMode为 EXACTLY,SpecSize为 400");
    Logger.d("main_btn测量尺寸, width: " + mMainBtn.getMeasuredWidth() + ", height: " + mMainBtn.getMeasuredHeight());
    Logger.d("main_btn实际尺寸, width: " + mMainBtn.getWidth() + ", height: " + mMainBtn.getHeight());

    //宽度设置SpecMode为AT_MOST,SpecSize为(1 << 30) - 1
    widthMeasureSpec = View.MeasureSpec.makeMeasureSpec((1 << 30) - 1, View.MeasureSpec.AT_MOST);
    heightMeasureSpec = View.MeasureSpec.makeMeasureSpec((1 << 30) - 1, View.MeasureSpec.AT_MOST);
    mMainBtn.measure(widthMeasureSpec, heightMeasureSpec);
    Logger.d("宽度设置SpecMode为 AT_MOST,SpecSize为 (1 << 30) - 1");
    Logger.d("main_btn测量尺寸, width: " + mMainBtn.getMeasuredWidth() + ", height: " + mMainBtn.getMeasuredHeight());
    Logger.d("main_btn实际尺寸, width: " + mMainBtn.getWidth() + ", height: " + mMainBtn.getHeight());

    //宽度设置SpecMode为UNSPECIFIED,SpecSize为0
    widthMeasureSpec = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED);
    heightMeasureSpec = View.MeasureSpec.makeMeasureSpec((1 << 30) - 1, View.MeasureSpec.AT_MOST);
    mMainBtn.measure(widthMeasureSpec, heightMeasureSpec);
    Logger.d("宽度设置SpecMode为 UNSPECIFIED,SpecSize为 0");
    Logger.d("main_btn测量尺寸, width: " + mMainBtn.getMeasuredWidth() + ", height: " + mMainBtn.getMeasuredHeight());
    Logger.d("main_btn实际尺寸, width: " + mMainBtn.getWidth() + ", height: " + mMainBtn.getHeight());
}

//log
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╔════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ Thread: main
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ MainActivity.onWindowFocusChanged  (MainActivity.java:93)
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║    MainActivity.testMeasureSpec  (MainActivity.java:62)
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ main_btn原始测量尺寸, width: 200, height: 96
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╚════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╔════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ Thread: main
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ MainActivity.onWindowFocusChanged  (MainActivity.java:93)
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║    MainActivity.testMeasureSpec  (MainActivity.java:67)
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ 宽度设置SpecMode为 EXACTLY,SpecSize为 400
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╚════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╔════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ Thread: main
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ MainActivity.onWindowFocusChanged  (MainActivity.java:93)
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║    MainActivity.testMeasureSpec  (MainActivity.java:68)
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ main_btn测量尺寸, width: 400, height: 96
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╚════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╔════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ Thread: main
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ MainActivity.onWindowFocusChanged  (MainActivity.java:93)
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║    MainActivity.testMeasureSpec  (MainActivity.java:69)
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ main_btn实际尺寸, width: 200, height: 96
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╚════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╔════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ Thread: main
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ MainActivity.onWindowFocusChanged  (MainActivity.java:93)
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║    MainActivity.testMeasureSpec  (MainActivity.java:75)
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ 宽度设置SpecMode为 AT_MOST,SpecSize为 (1 << 30) - 1
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╚════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.243 15885-15885/com.tomorrow.androidtest2 D/zwm: ╔════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ Thread: main
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ MainActivity.onWindowFocusChanged  (MainActivity.java:93)
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║    MainActivity.testMeasureSpec  (MainActivity.java:76)
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ main_btn测量尺寸, width: 176, height: 96
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╚════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╔════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ Thread: main
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ MainActivity.onWindowFocusChanged  (MainActivity.java:93)
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║    MainActivity.testMeasureSpec  (MainActivity.java:77)
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ main_btn实际尺寸, width: 200, height: 96
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╚════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╔════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ Thread: main
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ MainActivity.onWindowFocusChanged  (MainActivity.java:93)
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║    MainActivity.testMeasureSpec  (MainActivity.java:83)
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ 宽度设置SpecMode为 UNSPECIFIED,SpecSize为 0
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╚════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╔════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ Thread: main
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ MainActivity.onWindowFocusChanged  (MainActivity.java:93)
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║    MainActivity.testMeasureSpec  (MainActivity.java:84)
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ main_btn测量尺寸, width: 176, height: 96
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╚════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╔════════════════════════════════════════════════════════════════════════════════════════
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ Thread: main
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ MainActivity.onWindowFocusChanged  (MainActivity.java:93)
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║    MainActivity.testMeasureSpec  (MainActivity.java:85)
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╟────────────────────────────────────────────────────────────────────────────────────────
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ║ main_btn实际尺寸, width: 200, height: 96
03-25 12:58:55.253 15885-15885/com.tomorrow.androidtest2 D/zwm: ╚════════════════════════════════════════════════════════════════════════════════════════

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容