android解决无法设定listview的item高度

首先说明一下,在Android中,子控件的尺寸是由父控件决定的。

/*父控件的onLayout方法*/
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
    final int count = getChildCount();
    for (int i = 0; i < count; i++) {
        View child = getChildAt(i);
        if (child.getVisibility() != GONE) {
            RelativeLayout.LayoutParams st =
                    (RelativeLayout.LayoutParams) child.getLayoutParams();
            /*调用子类的layout方法,layout方法会继续调用onLayout方法*/
            child.layout(st.mLeft, st.mTop, st.mRight, st.mBottom);
        }
    }
}

从上面源码不仅可以看出子控件的尺寸由父控件所决定,而且还可以看出子控件的尺寸信息是存储在LayoutParams中的。
当我们构建了如下一个item.xml


    

然后在getView方法中通过以下代码inflate

View.inflate(RecycleViewActivity.this,R.layout.item,null)

虽然我们给item设置了100dp的高度和宽度,但是实际效果却是这样的:

android解决无法设定listview的item高度_第1张图片
Snip20161013_15.png

很明显,在这里设置宽高不起作用!
所以为了解决这个问题,百度以后,有下面两种方法

  • 在item的根布局中设置minHeight = "100dp",但是设置了minWidth = "100dp",也无法使item内容的宽度为100dp,原因后面会说到。
  • 再用一层父控件包裹,在第二层父控件设置android:layout_height = "100dp",android:layout_width = "100dp"布局如下:

    
        
    

以上两种"解决方法"只是有缺陷的投机取巧,因为minHeight和第二层父控件无法实现match_parent效果,所以这两种方法看一下就好。

文章开头说了,子控件的尺寸由父控件决定,并且存储在子控件的layoutParams属性中,下面看一下View.inflate()方法的源码

public static View inflate(Context context, @LayoutRes int resource, ViewGroup root) {
    LayoutInflater factory = LayoutInflater.from(context);
    return factory.inflate(resource, root);
}

public View inflate(@LayoutRes int resource, @Nullable ViewGroup root) {
    return inflate(resource, root, root != null);
}

public View inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot) {
    try {
        return inflate(parser, root, attachToRoot);
    } finally {
        parser.close();
    }
}

public View inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot) {
    View temp = createViewFromTag(root, name, inflaterContext, attrs);
    ViewGroup.LayoutParams params = null;

    //只有当root不为null的时候,才会创建layoutParams属性

    if (root != null) {
        params = root.generateLayoutParams(attrs);
        if (!attachToRoot) {
            // Set the layout params for temp if we are not
            // attaching. (If we are, we use addView, below)
            temp.setLayoutParams(params);    
        }
    }
}

由以上源码可以看出,由于我们调用View.inflate()方法传递的parent参数是null,所以导致item的layoutParams属性为null。所以真正解决无法设置ListView的Item的高度的方法是:使root != null

所以我们应该使用以下方法去inflate一个xml文件

View view = LayoutInflater.from(context).inflate(R.layout.item,parent,false);

到此为止,如何给ListView的Item设置高度的问题便解决了。


不过还有一个疑问,当我们使用错误的inflate方法的时候,为什么会呈现第一张图片的那种结果?

在AbsListView中有下面这三个方法

View obtainView(int position, boolean[] isScrap) {
    、、、
    setItemViewLayoutParams(child, position);
    、、、
}

private void setItemViewLayoutParams(View child, int position) {
    final ViewGroup.LayoutParams vlp = child.getLayoutParams();
    LayoutParams lp;
    if (vlp == null) {
        lp = (LayoutParams) generateDefaultLayoutParams();
    } else if (!checkLayoutParams(vlp)) {
        lp = (LayoutParams) generateLayoutParams(vlp);
    } else {
        lp = (LayoutParams) vlp;
    }
    if (mAdapterHasStableIds) {
        lp.itemId = mAdapter.getItemId(position);
    }
    lp.viewType = mAdapter.getItemViewType(position);
    if (lp != vlp) {
      child.setLayoutParams(lp);
    }
}

@Override
protected ViewGroup.LayoutParams generateDefaultLayoutParams() {
    return new AbsListView.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
            ViewGroup.LayoutParams.WRAP_CONTENT, 0);
}

通过代码可以看到,如果Item的layoutParams属性为空,那么默认为其设置一个宽度为match_parent,高度为wrap_content的layoutParams属性。所以如果是使用View.inflate(context,res,parent) inflate出来的view,其实其XML文件就相当于下面的例子:


    

由于layout_width = "match_parent",所以前面通过设置min_width = "100dp"是无法起作用的。


RecycleView的item也是类似的结果,只不过RecycleView在发现item的layoutParams == null的时候,默认生成的宽高约束都是wrap_content罢了。

你可能感兴趣的:(android解决无法设定listview的item高度)