Android开发中经常需要获取控件的宽高,比如前不久我在写一个图片加载库时,因为需要对Bitmap进行裁剪就遇到了需要获取ImageView宽高的问题。
如果稍微了解过一下View
的绘制过程,就会知道直接在onCreate()
等生命周期回调方法中获取宽高,获取到的值是0,所以我们需要采用一些特殊的方法去获取。
这里简单总结一下获取宽高的几种方法:
一、通过View.post(new Runnable())获取
这里的view可以是你需要获取宽高的View。要注意的是view
要执行此方法必须保证它已经attached到了window上,因此在此之前是不能调用这个方法的。
在Runnable
里可以执行获取宽高的方法:
view.post(new Runnable(){
@Override
public void run() {
int width = view.getMeasuredWidth();
int height = view.getMeasuredHeight();
}
});
这个方法的原理在于:View
的宽高需要在Measure
过程后才能确定,直接在onCreate()
等回调方法里获取只能得到0,因为此时还没有开始Measure
操作。
而通过view.post()
在主线程的消息队列尾部插入了一个消息,也就是说执行获取宽高的操作被延后了,并且能够保证Measure
操作在此之前,所以就能够在这里获取到正确的宽高了。
有关为什么“能够保证Measure
操作在此之前”,我会在下一篇文章里详细分析。
另外,在网上还可以搜到其他类似方法如使用
ViewTreeObserver.addOnGlobalLayout()/addOnPreDrawLayout()
或Activity/View.onWindowFocusChanged()
方法中获取的,本质也是延后了操作,等待View
初始化完毕了之后再进行获取,在这里就不细说了。
总结:
此方法优点是保证获取到的宽高是准确的;
缺点是不能及时获取到,实际上还是把操作延后了,需要在Runnable
里再执行相应回调。
二、通过LayoutParams获取
对于在XML
文件里设置了具体宽高的View
可以通过view.getLayoutParams().height/width
获取到宽高。
总结:优点是能及时获取到,且操作简单;缺点是不够通用,没有设置具体宽高的获取到的值就是0了。
三、手动Measure再获取
既然View
的宽高是在Measure
之后才能获得的,自行调用view.measure()
不就行了吗?
那么我们需要自行构造MeasureSpec
,也就是测量参数,它是一个由父容器和View
自身决定的变量。有关它的具体分析,可以阅读《Android开发艺术探索》来了解。
对于宽高设为具体数值或wrap_content
的控件,我们都可以手动构造MeasureSpec
,而match_parent
的情况理论上是做不到的。
对于设置了具体数值宽高的(比如都是100px),我们可以这样构造MeasureSpec
:
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_MOTST);
int heightMeasureSpec = MeasureSpec.makeMeasureSpec((1<<30)-1,MeasureSpec.AT_MOST);
view.measure(widthMeasureSpec,heightMeasureSpec);
总结:优点也是可以立即获取到宽高;缺点是无法解决match_parent
的情况。
网友评论
WindowManager wm = (WindowManager) getActivity()
.getSystemService(Context.WINDOW_SERVICE);
int width = wm.getDefaultDisplay().getWidth();
int w = View.MeasureSpec.makeMeasureSpec(0,
View.MeasureSpec.AT_MOST);
int h = View.MeasureSpec.makeMeasureSpec(0,
View.MeasureSpec.UNSPECIFIED);
// layoutSecLoan
layoutSecLoan.measure(w, h);
int layoutSecLoanW = layoutSecLoan.getMeasuredWidth();
//layout_rate
layout_rate.measure(w, h);
int layout_rateW = layout_rate.getMeasuredWidth();
LogMe.i(TAG, "SCREEN width:" + width + ", layoutSecLoan's Wdith:" + layoutSecLoanW + ",layout_rate's Wdith:" + layout_rateW);
}
然后layoutSecLoan's应该是整个item的宽,不知道为啥只有269,
SCREEN width:1280, layoutSecLoan's Wdith:269,layout_rate's Wdith:98