获取View的宽高

我们经常会需要获取View的宽度或者高度,但是我们会发现不管是在onCreate还是onStart,onResume里面获取到的值都是0.

这是因为View的measure过程和Activity的生命周期方法不是同步进行的,不能保证在Activity的onCreate,onStart,onResume方法执行时View已经测量完毕,所以不能获得正确宽高值。

因此我们需要通过其他方法来实现,如下几种方法都可以获取正确宽高值

  • Activity/View的onWindowFocusChanged方法体内
  • 利用view.post(runnable)
  • ViewTreeObserver
  • view.measure
  • 自定义View时获取宽高

Activity/View的onWindowFocusChanged方法体内

这个方法的含义是View已经初始化完毕了,自然可以获取宽高,但是该方法有可能多次执行,当Activity焦点变化的时候都会被调用。

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

view.post(runnable)

通过view的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.getMeasuredHeight();
        }
    });
}

这里在onCreate 和 onResume里面都可以。

ViewTreeObserver

顾名思义这个类是整个View树结构的观察者,通过它的众多回调可以获取到view的宽高,比如onGlobalLayoutListener接口,当View树的状体改变或者View树内部的View的可见性发生改变时该接口都将被回调,显然onGlobal也会被多次调用.

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

这里的removeOnGlobalLayoutListener方法需要API 16。

view.measure

该方法有两个参数即宽高的MeasureSpec,完整方法名为measure(int widthMeasureSpec,int heightMesureSpec) 如果对此不了解的可以先去学习关于view的measure过程.
通过手动对view测量得到宽高,这种方法较为复杂,要分情况处理,根据view的LayoutParams来分

  • match_parent
    根据View的measure过程分析构造此种MeasureSpec需要知道父容器的剩余空间,而此时无法知道,所以理论上不可能测出View的大小。
  • wrap_content
    如下
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);

int width = view.getMeasuredWidth());
int height = view.getMeasuredHeight());

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

另外关于View的measure,网络上有两个错误的用法:
1.

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

2.

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

自定义View时获取宽高

有时我们在自定义控件的时候会需要用到宽高值,大部分的时候可以在onLayout方法里面获取到真实宽高值,同样也可以用上述第三种方法获取。首先让自定义的View实现onGlobalListener接口,需要实现onGlobalLayout方法.

然后在onAttachedToWindow()添加

@Override
    protected void onAttachedToWindow() {
        super.onAttachedToWindow();
        getViewTreeObserver().addOnGlobalLayoutListener(this);
    }

记得要在onDetachedFromWindow()删除

@Override
    protected void onDetachedFromWindow() {
        super.onDetachedFromWindow();
        getViewTreeObserver().removeOnGlobalLayoutListener(this);(API 16)
    }

接着在onGlobalLayout方法中获取宽高即可,因为要调用多次,可以设立个标志位.

private boolean isFirst = true;
    @Override
    public void onGlobalLayout() {
        if (isFirst) {
            int height = getHeight();
            isFirst = false;
        }
    }