您的位置:首页 > 其它

Measure(0,0)到底发生了什么

2015-08-25 20:32 369 查看
最开始接触这个方法的时候是在写自定义控件的时候,比如,父控件的高度需要根据子控件来动态改变,那么我们就需要手动的去测量每一个子控件,然后他们的总高度就是父类最终的高度了。

在大多数情况下,这个逻辑可能是能够走通的,但是,我会举个栗子来让这个逻辑走不通。



1. 如果我对一个原生的View使用measure(0,0),是毫无意义的。

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),
getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));
}


如果大家点进getDefaultSize这个方法中仔细看的话,其结果还会是0,所以,你还是得不到其measuredWidth和measuredHeight。

但是,如果你对一个ImageView使用了该方法(前提是已经设置了图片给他),是有效果的。我通过断点的形式得到如下结果。

一起来看一看ImageView的onMeasure()方法吧。真正其变化的在这两行

w = Math.max(w, getSuggestedMinimumWidth());
h = Math.max(h, getSuggestedMinimumHeight());


所以真正使数据发生改变的就是这个getSuggestedMinimumWidth方法,点进去后看就会发现,你给ImageView设置了BackGround,那么他就会以这个背景来作为最小建议值,如果是原生的View的话,就是0.

后面的resolveSizeAndState方法是为了做更进一步的校正,即有可能图片的宽度比设备还大。等等。

2. 如果你给这个View设置了param,你还是测量不出,但是界面的显示会达到你预期的效果。

假设现在我们给这个View设置了param,即高度是200,宽度是-1(match_parent)。

如果此时我们去调用measure(0,0),仍然无法达到我们所要的效果,即得不到measuredWidth和measuredHeight,那么我们究竟应该怎么做呢,其实思路很简单。首先呢,measure()方法需要的是

widthMeasureSpec和heightMeasureSpec,如果我们还是给(0,0),说明我们队测量的过程是放任不管的

因为0代表public static final int UNSPECIFIED = 0。但是如果我们现在给他设置了param,说明我们现在是想管的,那既然想管,就得重新给他MeasureSpec,所以,我们自己make一个给他就行了。

注:MATCH_PARENT=-1 WRAP_CONTENT=-2;

private void measureView(View child) {
ViewGroup.LayoutParams p = child.getLayoutParams();
if (p == null) {
p = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT
,
ViewGroup.LayoutParams.WRAP_CONTENT);
}
int lpHeight = p.height;
int lpWidth = p.width;
int childHeightSpec;
int childWidthSpec;
if (lpHeight > 0) {   如果Height是一个定值,那么我们测量的时候就使用这个定值
childHeightSpec = MeasureSpec.makeMeasureSpec(lpHeight,
MeasureSpec.EXACTLY);
} else {   否则,我们将mode设置为不指定,size设置为0
childHeightSpec = MeasureSpec.makeMeasureSpec(0,
MeasureSpec.UNSPECIFIED);
}
if (lpWidth > 0) {
childWidthSpec= MeasureSpec.makeMeasureSpec(lpHeight,
MeasureSpec.EXACTLY);
} else {
childWidthSpec= MeasureSpec.makeMeasureSpec(0,
MeasureSpec.UNSPECIFIED);
}
child.measure(childWidthSpec, childHeightSpec);
}


如果我们使用上面的代码进行测量,param为(-1,200),那么我们得出的width=0,height=200.
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: