我们在开发中经常会遇到在界面中比如 activity 使用到控件的 width 或者 height,大家也都知道在 onCreate() 中去 getWidth() 或者是 getMeasuredWidth() 拿到的结果都是 0,这是因为我们的 activity 的创建和 view 绘制不是同步的,下面我总结了一些方法来获取 view 的宽度和高度的方法:

  • View.Post(new Runnable()) 方法
    这个方法并不是新启动一个线程,SDK 中对这个方法的注释为 Causes the Runnable to be added to the message queue. The runnable will be run on the user interface thread. 意思是在消息队列中添加一个 runnable ,这个 runnable 在用户界面线程上执行,也就是我们常说的安卓UI线程,那么我们将这个 runnable 插入到消息队列的末端,然后当 view 的 measure(),layout(),draw()执行完后调用我们在 runnable 中获取宽高的方法,就是准确的值了。缺点是方法不能及时执行。
  • Activity 的 onWindowFocusChanged() 方法
    当活动的当前窗口获得或失去焦点时调用。这是该活动是否对用户可见的最好指标。
    这个方法的含义是,view 已经初始化完毕了,这个时候取获取宽/高是没问题的。需要注意的是 onWindowFocusChanged 会被调用多次,当 Activity 的窗口得到焦点和失去焦点时均会被调用一次,就具体来说,当 activity 继续执行和暂停执行都会被调用,如果频繁地进行 onResume 和 onPause ,那么 onWindowFocusChange 也会频繁调用。
    代码如下:
@Override
    public void onWindowFocusChanged(boolean hasFocus) {
        super.onWindowFocusChanged(hasFocus);
        if (hasFocus) {
            int measuredHeight = view.getMeasuredHeight();
        }
    }
  • getViewTreeObserver().addOnGlobalLayoutListener()
    代码如下:
view.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                view.getViewTreeObserver().removeGlobalOnLayoutListener(this);
            }
        });
  • getViewTreeObserver().addOnPreDrawListener()
    代码如下:
view.getViewTreeObserver().addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
            @Override
            public boolean onPreDraw() {
                return false;
            }
        });
  • 通过 LayoutParams 获取
    对于在XML文件里设置了具体宽高的 View 可以通过view.getLayoutParams().height/width 获取到宽高。如果没有设置具体宽高的话就不能用这个方法了。
  • 通过 view 自身的测量方法 Measure()
    通过手动对 view 进行 measure 来得到宽高。这个方法得根据 view 的 LayoutParams 来区分:
    match_parent
    无法 measure 出具体的宽高。因为这个情况下,我们无法得到父容器的剩余空间,无法知道 parentSize ,所以不可能测量到 view 的大小,这个情况可以用上述其他方式来获取。
    具体的数值(dp/px)
int widthMeasureSpec = View.MeasureSpec.makeMeasureSpec(100, View.MeasureSpec.EXACTLY);
        int heightMeasureSpec = View.MeasureSpec.makeMeasureSpec(200, View.MeasureSpec.EXACTLY);
        view.measure(widthMeasureSpec, heightMeasureSpec);
**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);

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