Android在Activity中获取View宽高

在onCreate、onStart、onResume中获取不到View的宽高,因为View的measure过程和Activity的生命周期方法不是同步执行的,无法保证Activiy执行了onCreate、onStart、onResume时某个View已经完毕。因此想要在Activity中获取View的宽高需要用其他方式来解决:

Activity/View#onWindowFocusChanged

onWindowFocusChanged表示:View已经初始化完毕,宽/高已经测量好。
需要注意的是,当Activity的窗口得到焦点和失去焦点时均会调用onWindowFocusChanged。如果频繁地进行onResume和onPause,那么onWindowFocusChanged也会被频繁地调用。典型代码如下:

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

view.post(runnable)

通过post可以将一个runnable投递到消息队列,然后等到Lopper调用此runnable的时候,View也已经初始化好了,典型代码如下:

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

ViewTreeObserver

使用ViewTreeObserver的众多回调可以完成这个功能,比如使用OnGlobalLayoutListener这个接口,当View树的状态发生改变或者View树内部的View的可见性发生改变,onGlobalLayout方法就会回调,因此这是获取View的宽高一个很好的时机,需要注意的是,伴随着View树状态的改变,这个方法也会被调用多次,典型代码如下:

@Override
protected void onStart() {
    super.onStart();

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

view.measure(int widthMeasureSpec , int heightMeasureSpec)

通过手动测量View的宽高,需要根据View的LayoutParams来分情况处理

  • match_parent

无法测量出具体的宽高,根据View的测量过程,构造这种measureSpec需要知道parentSize,即父容器的剩下空间,而这个时候我们无法知道parentSize的大小,所以理论上我们不可能测量出View的大小

  • 具体的数值(dp/px)

比如宽高都是100px:

int widthMeasureSpec = MeasureSpec.makeMeasureSpec(100, MeasureSpec.EXACTLY);
int heightMeasureSpec = MeasureSpec.makeMeasureSpec(100, MeasureSpec.EXACTLY);
view.measure(widthMeasureSpec,heightMeasureSpec);
  • 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);

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

错误用法

关于View的measure,网络上有两个错误的用法,首先其违背了系统的内部实现规范(因为无法通过错误的MeasureSpec去得出合理的SpecMode,从而导致measure过程出错),其次不能保证mwasure出正确的结果

  • 第一种错误的方法:
int widthMeasureSpec = View.MeasureSpec.makeMeasureSpec(-1, View.MeasureSpec.UNSPECIFIED);
int heightMeasureSpec = View.MeasureSpec.makeMeasureSpec(-1, View.MeasureSpec.UNSPECIFIED);
mTextView.measure(widthMeasureSpec,heightMeasureSpec);
  • 第二种错误的用法:
mTextView.measure(LayoutParams.WRAP_CONTENT,LayoutParams.WRAP_CONTENT);