1.getItemViewType和getViewTypeCount

  getItemViewType和getViewTypeCount是ListView中实现复杂列表的两个相关的方法,普通的ListView中Item是相同的,那么我们只需要实现Adapter中四个抽象方法即可,但是如果页面中Item长得比较的复杂呢?比如说这个。

比如说这个列表项,其实也不是很复杂,这种类型的Item也有其他的实现方式,比如说在Adapter中实现SectionIndexer也是可以实现的,但是我们就拿这个来说明一下问题,如果一个Item第一种类型是TextView,第二种类型是ImageView+Button+TextView呢,那么这样复杂的列表我们就需要使用getItemViewType()和getTypeViewCount()两个方法去实现了。这两个方法理解起来还是比较容易的,获取Item中Type的类型以及Item中Type的相关数量。废话就不多说了,直接说实现方式。

public class ListAdapter extends BaseAdapter {/** * Item类型,int值.必须从0开始依次递增. * */ private static final int TYPE_TITLE = 0; private static final int TYPE_CONTENT = 1; /** * Item Type 的数量 * */ private static final int TYPE_ITEM_COUNT = 2; /** * 数据 * */ private List<Company> mData = new ArrayList<>(); private Context context; public ListAdapter(Context context,List<Company>mData){ this.context = context; this.mData = mData; } @Override public int getCount() { return mData.size(); } @Override public Object getItem(int position) { return mData.get(position); } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup viewGroup) { /** * 不同类型的ViewHolder * */ TitleViewHolder titleViewHolder = null; CompanyViewHolder contentViewHolder = null; /** * 对类型进行判断,分别inflate不同的布局. * */ switch (getItemViewType(position)){ case TYPE_TITLE: titleViewHolder = new TitleViewHolder(); if(convertView == null){ convertView = View.inflate(context, R.layout.view_holder_company_index,null); titleViewHolder.title = (TextView) convertView.findViewById(R.id.tv_title); //setTag()  convertView.setTag(titleViewHolder); }else{ //getTag(); titleViewHolder = (TitleViewHolder) convertView.getTag(); } titleViewHolder.title.setText(mData.get(position).getName()); break; case TYPE_CONTENT: contentViewHolder = new CompanyViewHolder(); if(convertView == null){ convertView = View.inflate(context,R.layout.view_holder_company,null); contentViewHolder.content = (TextView) convertView.findViewById(R.id.tv_content); convertView.setTag(contentViewHolder); }else{ contentViewHolder = (CompanyViewHolder) convertView.getTag(); } contentViewHolder.content.setText(mData.get(position).getCode()); break; } return convertView; } /** * 根据position获取Item的类型 * */ @Override public int getItemViewType(int position) { if(TextUtils.isEmpty(mData.get(position).getCode())){ return TYPE_TITLE; }else{ return TYPE_CONTENT; } } /** * 返回Item Type的总数量 * */ @Override public int getViewTypeCount() { return TYPE_ITEM_COUNT; } static class TitleViewHolder{ TextView title; } static class CompanyViewHolder{ TextView content; } }

  • 首先我们需要为不同的Item设置不同的数值,int值,因为getItemViewType返回的是int值,所以需定义成int,必须从0开始,依次递增。原因我后续会做出解释。
  • 重写getItemViewType和getViewTypeCount方法,getViewTypeCount返回Item的类型总数,getViewTypeCount则需要进行判断,判断方式一般都是通过JavaBean中的相关字段来判断的,因此这块不需要过于纠结。只需要根据position获取Item的具体类型进行判断然后就返回就可以了。
  • 定义ViewHolder,根据类型的不同需要定义多个ViewHolder,减少findViewById()的次数。
  • 重写getView()中的相关方法,在getView中首先根据position获取Item的类型去加载不用的布局,这里同时会setViewType为不同类型的Item设置RecycleBin,解决ListView由于多个类型Item的复用问题。不清楚RecycleBin机制的读者可以去看下ListView的复用机制这里说到了RecycleBin,如果不懂这个机制是看不明白下面的解释的。
  • 最后根据传递过来的数据setAdapter然后为Item进行赋值就完成了。

  大体的一个思路就是这样实现的,这里需要说一下为什么定义Item的类型的时候必须要从0开始,依次递增,那么原因是什么呢?如果我们有三种类型,我们将Item定义成1,2,4,那么势必会出现ArrayIndexOutOfBoundsException,也就是所谓的数组越界,我上网查了很多资料都说会出现异常,并且Google也确实标明了,Note: Integers must be in the range 0 to getViewTypeCount() - 1. IGNORE_ITEM_VIEW_TYPE can also be returned.但是看到这里就没有后续了,没人会去说明这个问题是怎样发生的,为什么要这样去定义。可能我就是闲的蛋疼的那种人,不弄明白确实感到不舒服。我在解释一下具体的原因:

  其实发生这种情况一般都是我们在下拉的时候出现的问题,在第一次加载第一页的时候是不会直接出现崩溃现象的,那么心细的读者可能会明白这有可能是ListView在复用时出现的问题,其实却是就是ListView复用机制导致的。我们来看一下这个方法:

/***ListView在针对不同Item复用时会调用这个方法*为每一种不同的Item设置一个RecycleBin,用于复用.*/
public void setViewTypeCount(int viewTypeCount) { if (viewTypeCount < 1) { throw new IllegalArgumentException("Can't have a viewTypeCount < 1"); } // noinspection unchecked /** * 根据viewTypeCount的数量设置一个ArrayList. * 同时为每一个Item再设置一个ArrayList,用来存储ScrapView. * 相当于一个二维数组来维护每一个Item的ScrapView数组. * 这里也就相当于为不同的Item设置单独的RecycleBin. */ ArrayList<View>[] scrapViews = new ArrayList[viewTypeCount]; for (int i = 0; i < viewTypeCount; i++) { scrapViews[i] = new ArrayList<View>(); } //保存ViewTypeCount,也就 = 2 mViewTypeCount = viewTypeCount; //当前的Scrap是第一个Item的ScrapView数组. mCurrentScrap = scrapViews[0]; //mScrapViews就保存了一个二维数组维护的RecycleBin. mScrapViews = scrapViews; }

这是具体的数据结构,简单理解就是每一个Item都有对应的ScrapView数组。这里其实并不是出问题的地方,我们都知道Item一旦被移出了屏幕,首先会Detach掉,然后被加入到mScrapView数组中(废弃View池),那么在addScrapView的时候就会出现异常.

  void addScrapView(View scrap) {  AbsListView.LayoutParams lp = (AbsListView.LayoutParams) scrap.getLayoutParams();  if (lp == null) { return; } // Don't put header or footer views or views that should be ignored // into the scrap heap int viewType = lp.viewType; if (!shouldRecycleViewType(viewType)) { if (viewType != ITEM_VIEW_TYPE_HEADER_OR_FOOTER) { removeDetachedView(scrap, false); } return; } /** *核心代码就是这块,由于我们mViewTypeCount != 1 的,因此if条件不成立. *因此会执行else代码. */ if (mViewTypeCount == 1) { dispatchFinishTemporaryDetach(scrap); mCurrentScrap.add(scrap); } else { dispatchFinishTemporaryDetach(scrap); mScrapViews[viewType].add(scrap); } if (mRecyclerListener != null) { mRecyclerListener.onMovedToScrapHeap(scrap); } } 

问题就在于这个else代码当中,我们可以看到mScrapViews[viewType].add(scrap)代码执行了,我们前面图片上显示的,mScrapViews[]是很据Item的种类数量new出来的,由于我们Item总数是两种类型,那么mScrapViews[].length = 2,但是这里是mScrapView[viewtype],viewtype是什么,其实就是我们getItemViewType的返回值,如果我们将类型定义成2和3,那么他会访问mScrapView[2]和mScrapView[3],可想而知,一定会出现ArrayIndexOutOffBoundsException.这就是数组越界的真正原因。



本文转自莫水千流博客园博客,原文链接:http://www.cnblogs.com/zhoug2020/p/6906120.html,如需转载请自行联系原作者

Android之ListView的getItemViewType和getViewTypeCount相关推荐

  1. Android一个ListView列表之中插入两种不同的数据

    http://www.cnblogs.com/roucheng/ Android一个ListView列表之中插入两种不同的数据 代码如下: public class ViewHolder{Button ...

  2. Android中ListView分类

    2019独角兽企业重金招聘Python工程师标准>>> 1. 引言 在Android开发过程中往往有这样的需求,将ListView中的内容按年,月,日进行分类显示,要实现这样的效果我 ...

  3. Android之ListView(三)

    文章目录 activity_main.xml chatting_item_msg_text_left.xml chatting_item_msg_text_right.xml ChatActivity ...

  4. android开发 listview 头部 轮播,listview添加的头部布局超过一屏头部内容显示不全...

    headView的实际高度超过一个屏幕,但是显示的结果只有一个屏幕,超过一个屏幕高度意外的部分显示不全. 只使用了listView.getRefreshable().addHeadView(headV ...

  5. Android使用ListView控件问题

    Android使用ListView控件问题: The application has stopped unexpectedly, please try again. 开发环境:android 1.6 ...

  6. android 分组 listview,Android实现的ListView分组布局改进示例

    本文实例讲述了android实现的listview分组布局改进方法.分享给大家供大家参考,具体如下: 由于是在网上转载的一篇文章,在这里就不多说废话了,首先看一下最终的效果图: 然后是实现该listv ...

  7. Android实现ListView异步加载图片

    转: http://www.iteye.com/topic/685986 ListView异步加载图片是非常实用的方法,凡是是要通过网络获取图片资源一般使用这种方法比较好,用户体验好,下面就说实现方法 ...

  8. Android 利用ListView制作带竖线的多彩表格

    1.listview与GridView 其实Android本身是有表格控件(GridView)的,但是GridView的每一列的宽度被限定为一样宽,有时设计表格时,列宽不可能为同一宽度,所有可以用Li ...

  9. Android DrawLayout + ListView 的使用(一)

    想做一个APP,设计中有侧边栏这个功能,所以现在开始学习下侧边栏的实现. 在官方的UI空间中已经给出了DrawerLayout这个侧滑的菜单空间. 因为在使用DrawerLayout的时候遇到了些问题 ...

最新文章

  1. AI也能写高考作文?我们用清华刚刚开源的「九歌」试了试
  2. asp调用打开exe文件
  3. 如何解决大规模机器学习的三大痛点?
  4. CAN波形解析实例(1)
  5. Linux Distribution Timeline for 2010(Linux 2010 年发行版时间线/族谱/发展图)
  6. python编程例子-python编程例子
  7. 2013北理机试题——中缀算术表达式对应二叉树的先序遍历
  8. VS生成dump文件和调试dump文件
  9. Linux查看服务器自动关机,Linux系统自动关机的命令详解
  10. Linux基础知识、常用命令和操作
  11. Markdown 下载安装
  12. 扫码枪 android EditText
  13. [架构之路-51]:架构师 - 用系统化、结构化思维解决复杂难搞的软件故障问题 - 马克思主义哲学在软件系统中的应用
  14. 基于Java发送邮件
  15. C语言实现通过日期计算这是一年中的第几天
  16. ROS Spinning-----简介
  17. 基于多阈值的形态提取遥感图像中的沿海线的特征方法(Qu Jishuang)
  18. 微信摇一摇php,微信摇一摇功能实现 - 微信公众平台开发:微信
  19. 读书笔记:收敛性 ← 随机过程
  20. vuforia for unity 入门教程

热门文章

  1. Spring MVC快速教程:依赖注入 Spring MVC Fast Tutorial: Dependency Injection
  2. 大学生如何进化到程序猿
  3. 关于asp.net中链接数据库的问题
  4. 《告别失控:软件开发团队管理必读》一一1.2 成功的程序设计经理为什么难当...
  5. 10-9-堆排序-内部排序-第10章-《数据结构》课本源码-严蔚敏吴伟民版
  6. linux下ftp服务器的搭建与使用
  7. 3.5 mysql备份与恢复
  8. Django 运行 runserver 端口占用,报错:Errno 10013
  9. Column store index 列数据如何匹配成行数据?
  10. 将不确定变为确定~整形变量是否可以进行位运算(像枚举类型一样)