我们在开发过程中经常会遇到在activity启动时就需要获取view的高度和宽度,我们发现在oncreate、onresume,onstart这几个方法中都无法正确的获取view的高度和宽度。很多时候获取到的值都是0 。这让我头疼了很久,查看资料,以及源码分析,终于找到了解决的办法,在下边和小伙伴们分享,分享。
(1)activity/view#onwindowfocuschanged 这个方法的含义是:view已经初始化完毕了,高和宽已经准备好了,这个时候去获取高和宽是没有问题的。需要注意的是onwindowfocuschanged 这个方法会被调用多次,当activity窗口获得焦点和失去焦点时均会被调用一次。具体来说当activity继续执行和暂停执行时均会被调用一次,如果平凡的进行onresume和onpause,onwindowfocuschanged 将会被调用多次。典型代码如下:
@Override
public void onWindowFocusChanged(boolean hasFocus) {
super.onWindowFocusChanged(hasFocus);
if (hasFocus) {
int width = view.getMeasuredWidth();
int height = view.getMeasuredHeight();
}
}
(2)view.post(runnable)
通过post可以将一个runnable投递到消息队列的尾部,然后等待looper调用此方法的时候,view的初始化也已经完成了。典型代码如下:
@Override
protected void onStart() {
super.onStart();
view.post(new Runnable() {
@Override
public void run() {
int width = view.getMeasuredWidth();
int height = view.getMeasuredHeight();
}
});
}
(3)viewtreeobserver
使用viewtreeobserver的众多回调可以完成这个功能,比如使用ongloballayoutlistener这个借口,当view的状态发生改变,或者view树内部的view就可见性发生变化时,ongloblayout会被调用多次,典型代码如下:
@Override
protected void onStart() {
super.onStart();
ViewTreeObserver vObserver = view.getViewTreeObserver();
vObserver.addOnGlobalLayoutListener(new OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
view.getViewTreeObserver().removeOnGlobalLayoutListener(this);
int width = view.getMeasuredWidth();
int height = view.getMeasuredHeight();
}
});
}
(4)view.measure(int widthmeasurespec,int heightmeasurespec)
通过手动对view进行measure来得到view的高和宽,这个方法比较复杂,这里面要分情况,根据view的layoutpaeams来划分:
match_parent 直接放弃,无法measure出view的具体高度,通过measure的分析可以知道,这个时候需要知道parentsize才能计算出高和宽,就是父容器的高和宽,但是我们无法得到。
具体数值(dp/px) 比如宽高都是100px
int measurewidth = MeasureSpec
.makeMeasureSpec(100, MeasureSpec.EXACTLY);
int measureheight = MeasureSpec.makeMeasureSpec(100,
MeasureSpec.EXACTLY);
view.measure(measurewidth, measureheight);
wrap_content
int measurewidth = MeasureSpec.makeMeasureSpec((1 << 30) - 1,
MeasureSpec.AT_MOST);
int measureheight = MeasureSpec.makeMeasureSpec((1 << 30) - 1,
MeasureSpec.AT_MOST);
view.measure(measurewidth, measureheight);
注意(1《30)-1,通过分析measurespec可以知道view的尺寸使用30位二进制表示,也就是说最大30个1,也就是(1《30)-1,在最大模式下,我们的view宁支持的最大值去构造measurespec是合理的。
关于view的measure,网络上有两个错误的用法,因为他违背了系统内部实现规范,不能保证一定能measure出正确结果。
第一种错误:
int measurewidth = MeasureSpec.makeMeasureSpec( - 1,
MeasureSpec.AT_MOST);
int measureheight = MeasureSpec.makeMeasureSpec( - 1,
MeasureSpec.AT_MOST);
view.measure(measurewidth, measureheight);
第二种:
view.measure(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
以上是我对再启动activity时就获取view的宽和高的一些总结,希望对你有用