Android ListView 进阶学习

时间:2023-12-05 13:50:56

1.使用ListView展示数据结构为二维数组的数据
当我们遇到数据结构是二维数组的需求的时候,我们会首先想到ListView,但是要想实现二维数组,会想到ListView里面嵌套ListView,但是显然这样是不可以的,这个时候推荐使用BaseExpandableListAdapter,这个需求UI原型可以参考QQ的好友及分组的UI,这样就能够了然。

2.ListView存在getView缓存的问题
解决方案是在Adapter的getView的时候使用ViewHolder,不仅能够缩短findViewById()的次数来增加效率,而且还能在展示的时候准确的展示内容。

3.ListView的自动定位到锚点
ListView通过smoothScrollToPositionFromTop 来进行自动移动到指定位置。
ExpandableListView 是继承的ListView, 所以也可以这样做,但是在这里需要注明的是group和Child都算是一个position。

4.addHeaderView 和 addFooterView 与 setAdapter的先后顺序的问题:

最好在setAdapter的方法之前调用到,否则在4.3及以下的手机上出现问题。

下面是针对此问题的说明:
ListView想要添加headerview的话,就要通过addHeaderView这个方法,然后想要为ListView设置数据的话,就要调用setAdapter方法了。但是,在调用addHeaderView和setAdapter的顺序上,有时会爆出 java.lang.IllegalStateException: Cannot add header view to list -- setAdapter has already been called.的异常。这是因为我们在addHeaderView之前调用了setAdapter。所以,在这里,建议setAdapter需要在addHeaderView和addfooterView之后调用。这样就安全了。下面,我们来看看源码吧。看看究竟是什么原因造成的。
Android-18(4.3)的addHeaderView源码:

publicvoid addHeaderView(View v, Object data, boolean isSelectable) {
    final FixedViewInfo info = new FixedViewInfo();
    info.view = v;
    info.data = data;
    info.isSelectable = isSelectable;
    mHeaderViewInfos.add(info);
    if (mAdapter != null) {
        if (!(mAdapter instanceof HeaderViewListAdapter)) {
            mAdapter = new HeaderViewListAdapter(mHeaderViewInfos, mFooterViewInfos, mAdapter);
        }
        if (mDataSetObserver != null) {
            mDataSetObserver.onChanged();
        }
    }
}

Android-17(4.2)的addHeaderView的源码:

public void addHeaderView(View v, Object data, boolean isSelectable) {
    if (mAdapter != null && ! (mAdapter instanceof HeaderViewListAdapter)) {
        thrownew IllegalStateException("Cannot add header view to list -- setAdapter has already been called.");
    }
    FixedViewInfo info = new FixedViewInfo();
    info.view = v;
    info.data = data;
    info.isSelectable = isSelectable;
    mHeaderViewInfos.add(info);
    if (mAdapter != null && mDataSetObserver != null) {
        mDataSetObserver.onChanged();
    }
}

在上面,我们可以对比出代码中的处理的不同。在17版本中,只要adapter不为空的话,那就直接会抛出异常,而这个异常恰好就是我们文章开头说到的异常。在18版本中,如果adapter不为空的话,则会新建一个adapter,这个adapter会包含了headerview和footerview以及我们传进来的原来的adapter。这是在18版本以后做的一个处理。虽然有了处理,但是建议大家还是按照上面说的调用顺序来使用addHeaderView,addFooterView和setAdapter吧。