Android中分类的ListView实现

平时开发中我们中能遇到一种列表页面但是确实两种或者更多分类的Item布局才能实现,比如手机通讯录界面。在最近代码优化中发现了自己应用中的几个界面中也是类似的UI,但是早期的实现方式是一个Item把所有布局都写上去然后控制来显示,无形之中某些Item多分配了不必要的内存。

如下面这张UI效果图:
这里写图片描述
这是我们应用中的一处UI效果图,早期为了赶进度确实用了一种最”无奈”的实现方式,就是上面所说的全部都布局出来然后控制显示。
我们可以仔细观察图片发现可以分三块来实现,相关用户跟教程一块,用户,教程这三块。那么如何来实现呢?

1.分别布局了title.xml, user.xml, course.xml三个布局。
2.定义一个BaseAdapter如下

public class ReCommondTutorialAdapter extends BaseAdapter {@Overridepublic int getCount() {

    return tutorials == null ? 0 : tutorials.size();

}


@Overridepublic Object getItem(int position) {

    return tutorials.get(position);

}


@Overridepublic long getItemId(int position) {

    return position;

}


@Overridepublic int getViewTypeCount() {

    return 0;

}


@Overridepublic int getItemViewType(int position) {

    return 0;  
}



  @Overridepublic View getView(int position, View view, ViewGroup parent) {

    return view;

}


其中主要是要重写

public int getItemViewType(int position) {
    return 0;// 用于返回当前的item是应该属于哪一个type从而显示哪一个布局
}

public int getViewTypeCount() {
    return 1;// 返回一个有多少个布局
}

这样按照具体的业务需求实现,下面是我代码中的代码片段:

public class ReCommondTutorialAdapter extends BaseAdapter {


private static final int ITEMTYPECOUNT = 3;

private static final int TYPE_USER = 0;

private static final int TYPE_COURSE = 1;
private static final int TYPE_TITLE = 2;


@Overridepublic int getCount() {

    return tutorials == null ? 0 : tutorials.size();

}


@Overridepublic Object getItem(int position) {

    return tutorials.get(position);

}


@Overridepublic long getItemId(int position) {

    return position;

}


@Overridepublic int getViewTypeCount() {

    return ITEMTYPECOUNT;

}


@Overridepublic int getItemViewType(int position) {
             TutorialListInfo _tutorial = tutorials.get(position);
                  if (_tutorial.isUser()) {

    return TYPE_USER;// 返回的是用户列表

} else if (_tutorial.isTitle) {
    return TYPE_TITLE;// 返回标题
}

}



@Overridepublic View getView(int position, View view, ViewGroup parent) {

final TutorialListInfo info = tutorials.get(position);
        int type = getItemViewType(position);

ViewHolder holder;

switch (type) {

    case TYPE_COURSE:

    if (null == view) {

            view =LayoutInflater.from(mContext).inflate(xx.xml, null);

holder = new ViewHolder();
                    view.setTag(holder);

} else {

    holder = (ViewHolder) view.getTag();

}

// xxx业务代码break;

case TYPE_USER:

if (null == view) {

view = LayoutInflater.from(mContext).inflate(xx.xml, null);

holder = new ViewHolder();
                    view.setTag(holder);

} else {

    holder = (ViewHolder) view.getTag();

}

// xxx业务代码break;
case TYPE_TITLE:
// xxx一样的标题
break;

return view;

}
}


这样就可以了,但是我们中间需要注意一点是:
每个item对应的int标识不能超过count的。也就是说如果你有两个布局文件,count就是2,每个布局文件对应的int值只能从0,1中间选择,不能超过2,不让就会报错。

public void setViewTypeCount(int viewTypeCount) {
    if (viewTypeCount < 1) {
      throw new IllegalArgumentException("Can't have a viewTypeCount < 1");
    }
    //noinspection unchecked
    SparseArray<View>[] scrapViews = new SparseArray[viewTypeCount];
    for (int i = 0; i < viewTypeCount; i++) {
      scrapViews[i] = new SparseArray<View>();
    }
    this.viewTypeCount = viewTypeCount;
    currentScrapViews = scrapViews[0];
    this.scrapViews = scrapViews;
  }

上面是一段ListView的缓存机制中代码片段,当多个type的时候就会存在多个SparseArray来缓存相对应的Views,因为ListView中存在缓存机制,会根据不同的ItemType缓存不同的item对象,这样再次滑动到相同的item时就能从中取出来使用无需再new一个新的出来,这样也解决了之前的代码中”蠢”的实现方式,优化之后感觉整体滑动流畅了,内存占用也减少了。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值