Question
在Activity 已启动的时候需要获取某个View 宽/高?
Analysis
通常操作会在onCreate或者onResume里面去获取这个View的宽/高,但是实际上无法正确获得某个View的宽/高的信息。这是因为View 的measure过程和Activity的生命周期方法不是同步执行的,因此无法保证Activity执行了onCreate、onResume时某个View已经测量完毕了,如果View还没有测量完毕了,那么获得 宽/高就是0.
Resolve
- Activity/View.onWindowFocusChanged
该方法的含义是:View 已经初始化完毕了,宽/高已经准备好了,这个时候去获取宽/高是没问题的。注意: onWindowFocusChanged会被调用多次,当Activity 的窗口得到焦点和失去焦点时均会被调用一次。具体来说,如果频繁地进行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投递到消息队列的尾部,然后等待Looper 调用此runnnable 的时候,View已经初始化好了。
view.post(
new Runnable(){
public void run(){
int width=view.getMeasuredWidth();
int height=vieew.getMeasuredHeight();
}
}
);
- ViewTreeObserver
使用ViewTreeObserver 的众多回调可以完成这个功能,比如使用OnGlobalLayoutListener 这个接口,当View 树的状态发生改变或者View 树内部的View的可见性发生改变时,onGlobalLayout 方法将被回调,因此这是获取View的宽/高一个很好的时机。注意: 伴随着View 树的状态改变等,onLayoutGlobal 会被调用多次。
ViewTreeObserver observer=view.getViewTreeObserver();
observer.addOnGlobalLayoutListener(new OnGlobalLayoutListener(){
public void onGlobalLayout(){
view.getViewTreeObserver().removeGlobalOnLayoutListener(this);
int width=view.getMeasuredWidth();
int height=vieew.getMeasuredHeight();
}
});
- view.measure(int widthMeasureSpec, int heightMeasureSpec)
通过手动对View 进行measure来得到View 的宽/高。这里要分情况处理,根据View的LayoutParams来分:
- match_parent
这种情况无法measure出具体的高度。原因是:根据View 的measure过程,构造此种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);
View 的尺寸使用30位二进制表示,也就是说最大是30个1(即2^30-1),也就是(1<<30)-1,在最大化模式下,用View 理论上能支持的最大值去构造MeasureSpec是合理的。