Android获取View宽/高的四种方式

1.Activity/View#onWindowFocusChanged

示例代码如下:

public void onWindowFocusChanged(boolean hasFocus) {
        super.onWindowFocusChanged(hasFocus);
        if (hasFocus) {
            int width = mView.getMeasuredWidth();
            int height = mView.getMeasuredHeight();
        }
    }```
注意:此方法会被多次调用,当Activity的窗口得到/失去焦点时均会被调用一次
#### 2.view.post(runnable)
示例代码:

mView.post(new Runnable() {
@Override
public void run() {
int width = mView.getMeasuredWidth();
int height = mView.getMeasuredHeight();
}
});```

3.ViewTreeObserver

示例代码如下:
监听OnGlobalLayoutListener事件

ViewTreeObserver mViewTreeObserver = mView.getViewTreeObserver();
        mViewTreeObserver.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                mView.getViewTreeObserver().removeOnGlobalLayoutListener(this);
                int width = mView.getMeasuredWidth();
                int height = mView.getMeasuredHeight(); 
       }
});```
监听OnPreDrawListener事件

ViewTreeObserver mViewTreeObserver = mView.getViewTreeObserver();
mViewTreeObserver.addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
@Override
public boolean onPreDraw() {
mView.getViewTreeObserver().removeOnPreDrawListener(this);
int width = mView.getMeasuredWidth();
int height = mView.getMeasuredHeight();
return false;
}
});

注意:伴随着View树的状态变化等,onGlobalLayout会被调用多次
#### 4.view.measure(int widthMeasureSpec, int heightMeasureSpec)(不推荐)
通过手动对View进行measure来得到View的宽/高。这种方法比较复杂,这里要分情况处理,根据View的LayoutParams来分:
###### match_parent
直接放弃,无法measure出具体的宽/高。原因很简单,根据View的measure过程,构造此种MeasureSpec需要知道parentSize,即父容器的剩余空间,而此时我们无法知道parentSize的大小,所以理论上不可能测量出View的大小。
###### 具体的数值(dp/px)
比如宽/高都是100px,如下measure:

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

###### 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);
mView.measure(widthMeasureSpec, heightMeasureSpec);

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

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

int widthMeasureSpec = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED);
int heightMeasureSpec = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED);
mView.measure(widthMeasureSpec, heightMeasureSpec);

第二种错误用法:

mView.measure(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);```
相关文章:
Github项目解析(八)-->Activity启动过程中获取组件宽高的五种方式
Android开发之getMeasuredWidth和getWidth区别从源码分析

你可能感兴趣的:(Android获取View宽/高的四种方式)