Activity中获取某个View宽高信息的四种方法

766 阅读2分钟

转载:【android】获取某个View宽高信息的四种方法

如果我们想在activity已启动时去获取某个View的宽、高,实际上在onCreate,onStart,onResume中均无法正确得到某个View的宽高信息。因为View的measure过程和activity的生命周期方法不是同步执行的,因此无法保证Activity执行了onCreate,onStart,onResume时,某个View已经测量完毕,一旦View没有测量完毕,那么我们此时获得的宽/高就是0。

解决的四种方法

1,onWindowFocusChanged

该方法的含义是:View已经初始化完毕了,宽/高已经准备好了,所以此时去获取宽/高是没有问题的。

注意:onWindowFocusChanged会被调用多次,当activity的窗口得到焦点和失去焦点时均会被调用一次,具体来说,当activity继续执行(onResume)和暂停执行(onPause)时,onWindowFocusChanged均会被调用。

@Override
    public void onWindowFocusChanged(boolean hasFocus) {
        super.onWindowFocusChanged(hasFocus);
        if (hasFocus) {
            int width = view.getMeasuredWidth();
            int height = view.getMeasuredHeight();
        }
    }

2,view.post(runnable)

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

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

3,ViewTreeObserver

使用ViewTreeObserver的众多回调也可以完成这个功能,比如使用OnGlobalLayoutListener这个接口。当View树的状态发生改变或者View树内部的View的可见性发生改变时,onGlobalLayout方法将被调用。

注意:伴随着View树的状态改变等,onGlobalLayout会被调用多次。因此需要在适当时机将监听回调移除。

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

4,view.measure(int widthMeasureSpec, int heightMeasureSpec)

手动对View进行measure来得到View的宽/高,这种方法比较复杂,我们还需要分情况处理,根据View的LayoutParams来区分。

(1) match_parent

直接放弃,无法measure出具体的宽/高。因为在View的measure过程中,构造这种MeasureSpec需要知道parentSize(父容器的剩余空间),此时我们是无法得知parentSize的大小,所以理论上是不可能测量出View的大小。

(2) 具体的数值(dp/px)

比如宽高都是50px,如下measure:

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

(3) wrap_content

如下measure:

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

(1 << 30) - 1,这个数值。分析下MeasureSpec的实现可知道,View的尺寸使用30位二进制表示,也就是说最大是30个1(2^30-1),也就是(1 << 30) - 1,在最大化模式下,我们用View理论上能支持的最大值去构造MeasureSpec是合理的。

两种错误的用法示例:

int widthMeasureSpec = View.MeasureSpec.makeMeasureSpec(-1,View.MeasureSpec.UNSPECIFIED);
int heightMeasureSpec = View.MeasureSpec.makeMeasureSpec(-1,View.MeasureSpec.UNSPECIFIED);
view.measure(widthMeasureSpec, heightMeasureSpec);

view.measure(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);

为什么说这两种是错误的,首先其违背了系统的内部实现规范(无法通过错误的MeasureSpec去得出合法的MeasureSpec,从而导致measure过程出错),其次不能保证一定能measure出正确的结果。