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

最开始接触这个方法的时候是在写自定义控件的时候,比如,父控件的高度需要根据子控件来动态改变,那么我们就需要手动的去测量每一个子控件,然后他们的总高度就是父类最终的高度了。
在大多数情况下,这个逻辑可能是能够走通的,但是,我会举个栗子来让这个逻辑走不通。
Measure(0,0)到底发生了什么_第1张图片
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.

你可能感兴趣的:(Android开发)