ListView往往可能会有不同的数据类型,单类型的数据可能运用会比较少些,这也是最近项目中的一个需求{在发送消息的时候,需要选择联系人,而联系人列表由英文字母索引+联系人组成},上一篇文章只是一个基调,这篇是更复杂的情况;
先看一下效果图

最开始的时候,打算把两种数据类型放入一个List<Object>中,参考上一篇随笔的状态保持的实现,在代码写完了开始测试的时候,发现问题众多,上下滚动的时候左边的CheckBox的选择状态没有很好的保存,会出现混乱选择的情况,于是参考网上的一些做法{寻找的参考方法并没有描述像这样稍稍复杂点的情况,都是TextView,没有状态的保持,没有View的重用,所以写了这篇随笔}并延伸总结;
MutiTypeAdapter.java

public class MutiTypeAdapter extends BaseAdapter {
private OnSelectedItemChanged listener;
private List<ListItem> list;
private LayoutInflater inflater; public MutiTypeAdapter(Context context, List<ListItem> list,
OnSelectedItemChanged listener) {
super();
this.list = list;
inflater = LayoutInflater.from(context);
this.listener = listener;
} @Override
public int getCount() {
return list.size();
} @Override
public Object getItem(int position) {
return list.get(position);
} @Override
public long getItemId(int position) {
return position;
} @Override
public View getView(int position, View convertView, ViewGroup parent) {
// 重点
View view = list.get(position).getView(convertView, inflater);
if (list.get(position).getClass() == BEntity.class) { // 如果是BEntity,也就是上面图中左边有CheckBox的项
final BEntity entity = (BEntity) list.get(position);
final CheckBox cb = entity.cbox;
cb.setChecked(entity.isChecked());
cb.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
entity.setChecked(cb.isChecked()); // 更改List中Entity的选择状态
if (listener != null) {
listener.onClick(getSelectedItem(list)); // 接口的思想暴露给Activity选择了多少项,当然也可以具体点通知Activity选择了哪些项
}
}
});
}
return view;
} public int getSelectedItem(List<ListItem> list) { // 获取选择了多少项
int i = 0;
for (ListItem item : list) {
if (item.isChecked()) {
i++;
}
}
return i;
} public interface OnSelectedItemChanged {
public void onClick(int count);
}
}

上面是数据源适配器,最开始的时候我在getView方法中对Item进行数据类的判断(AEntity/BEntity),再决定是选择加载哪一个layout,结果发现在重用View的时候很混乱,所以改为上面的实现方法;
AEntity和BEntity都继承自接口ListItem

public class AEntity implements ListItem {
private String str; public AEntity(String str) {
super();
this.str = str;
} @Override
public View getView(View convertView, LayoutInflater inflater) {
Holder holder = null;
if (convertView == null
|| convertView.getTag().getClass() != Holder.class) {
holder = new Holder();
convertView = inflater.inflate(getLayoutId(), null);
TextView tv = (TextView) convertView.findViewById(R.id.title_tv);
holder.tv = tv;
convertView.setTag(holder);
} else {
holder = (Holder) convertView.getTag();
}
holder.tv.setText(str);
return convertView;
} class Holder {
TextView tv;
} @Override
public int getLayoutId() {
return R.layout.title;
} @Override
public boolean isChecked() { // 此Entity相当于是标题项,没有CheckBox,所以永远返回false
return false;
}
}
public class BEntity implements ListItem {
private boolean isChecked = false;
private String str; public boolean isChecked() {
return isChecked;
} public void setChecked(boolean isChecked) {
this.isChecked = isChecked;
} public BEntity(String str) {
super();
this.str = str;
} @Override
public int getLayoutId() {
return R.layout.child;
} public CheckBox cbox; @Override
public View getView(View convertView, LayoutInflater inflater) {
Holder holder = null;
if (convertView == null
|| convertView.getTag().getClass() != Holder.class) {
holder = new Holder();
convertView = inflater.inflate(getLayoutId(), null);
TextView tv = (TextView) convertView.findViewById(R.id.item_tv);
CheckBox cb = (CheckBox) convertView.findViewById(R.id.item_cb);
holder.tv = tv;
holder.cb = cb;
convertView.setTag(holder);
} else {
holder = (Holder) convertView.getTag();
}
holder.tv.setText(str);
final CheckBox cb = holder.cb;
this.cbox = cb;return convertView;
} class Holder {
TextView tv;
CheckBox cb;
}
}

ListItem.java

public interface ListItem {
public boolean isChecked(); // 当前项是否选中 public int getLayoutId(); public View getView(View convertView, LayoutInflater inflater); // 返回Adapter中需要返回的View
}

在MainActivity中,模拟数据源并绑定到ListView列表;

public class MainActivity extends Activity {
ListView lv;
MutiTypeAdapter adapter; @Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
lv = (ListView) findViewById(R.id.lv);
setAdapter();
} private void setAdapter() {
List<ListItem> list = new ArrayList<ListItem>();
for (int i = 0; i < 50; i++) {
if (i % 2 == 0) {
list.add(new AEntity("item - " + i));
} else {
list.add(new BEntity("item - " + i));
}
}
OnSelectedItemChanged listener = new OnSelectedItemChanged() { @Override
public void onClick(int count) {
Log.e("SelectedCount", count + "");
}
};
adapter = new MutiTypeAdapter(getApplicationContext(), list, listener);
lv.setAdapter(adapter);
}
}

OK,通过几步就实现了所想要的功能!

Android ListView 显示多种数据类型的更多相关文章

  1. Android进阶笔记11:ListView篇之ListView显示多种类型的条目(item)

    ListView可以显示多种类型的条目布局,这里写显示两种布局的情况,其他类似. 1. 这是MainActivity,MainActivity的布局就是一个ListView,太简单了这里就不写了,直接 ...

  2. Android进阶笔记10:ListView篇之ListView显示多种类型的条目(item)

    ListView可以显示多种类型的条目布局,这里写显示两种布局的情况,其他类似. 1. 这是MainActivity,MainActivity的布局就是一个ListView,太简单了这里就不写了,直接 ...

  3. ListView显示多种类型的item

    ListView可以显示多种类型的条目布局,这里写显示两种布局的情况,其他类似 这是MainActivity:,MainActivity的布局就是一个ListView public class Mai ...

  4. Android ListView显示不同样式的item

    先look图 我们再使用listview时,listview的item大多时候都是一种样式,在很多app中也很常见,但有时候根据需求,可能数据的数量不一样,同个类型的数据显示的位置不同,亦或者有的it ...

  5. Android ListView显示底部的分割线

    有些时候,我们会提出这样的需求,希望ListView显示底部(顶部)的分割线,这样做,会使得UI效果更加精致,如下图所示: 如果搜索资料,大家会搜到一堆相关的方法,最多的莫过于设置listview的f ...

  6. Android ListView添加多种类型的ItemView

    一般复杂的ListView都会重写BaseAdapter,通过重用convertView来减少inflate,通过setTag()和ViewHolder改变ItemView的内容. 重写BaseAda ...

  7. Android ListView显示访问WebServices返回的JSON结果

    1.WebServices的返回结果 2.ListView内容布局代码 <?xml version="1.0" encoding="utf-8"?> ...

  8. Android ListView滑动过程中图片显示重复错乱闪烁问题解决

    最新内容建议直接访问原文:Android ListView滑动过程中图片显示重复错乱闪烁问题解决 主要分析Android ListView滚动过程中图片显示重复.错乱.闪烁的原因及解决方法,顺带提及L ...

  9. Android ListView分页加载时图片显示问题

    场景:Android ListView需要分页加载,每个item中会有图片,图片又是从网络下载的. 问题:在滑动加载下一页时,上一页的图片明明已经下载完成了,但是无法显示出来. Bug重现: 1,加载 ...

随机推荐

  1. 05-JVM对象探秘

    一.对象的内存布局         以Hotspot虚拟机为例,对象在内存中的结构可以分为三部分:对象头(header).实例数据(instance data).对齐填充(padding). 1.1. ...

  2. hdu2544最短路(floyd基础)

    最短路 Time Limit: 5000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submiss ...

  3. 第六模块:WEB框架开发 第1章·Django框架开发1~50

    01-Django基础介绍 02-Web应用程序1 03-Web应用程序2 04-http请求协议1 05-http请求协议2 06-http协议之响应协议 07-wsgire模块1 08-wsgir ...

  4. Python汉诺塔问题递归算法与程序

    汉诺塔问题: 问题来源:汉诺塔来源于印度传说的一个故事,上帝创造世界时作了三根金刚石柱子,在一根柱子上从上往下从小到大顺序摞着64片黄金圆盘.上帝命令婆罗门把圆盘从下面开始按大小顺序重新摆放在另一根柱 ...

  5. 三个线程ABC,交替打印ABC

    转载与:https://www.cnblogs.com/x_wukong/p/4009709.html 创建3个线程,让其交替打印ABC . 输出如下:  ABCABCABCABC. 方法:使用syn ...

  6. 什么是Spark

    什么是Spark Apache Spark是一个开源集群运算框架, 相对于Hadoop的MapReduce会在运行完工作后将中介数据存放到磁盘中,Spark使用了存储器内运算技术,能在数据尚未写入硬盘 ...

  7. Redis4.0支持的新功能说明

    本文以华为云DCS for Redis版本为例,介绍Redis4.0的新功能.文章转载自华为云帮助中心. 与Redis3.x版本相比,DCS的Redis4.x以上版本,除了开源Redis增加的特性之外 ...

  8. 详解Jedis连接池报错处理

    在使用Jedis连接池模式下,比较常见的报错如下: redis.clients.jedis.exceptions.JedisConnectionException:Could not get a re ...

  9. CVPR-2018 那些有趣的新想法

    Taylor Guo @ Shanghai - 2018.10.18 缘起 还有什么比顶级会议更适合寻找有趣新想法的地方吗?我们从CVPR 2018 计算机视觉和模式识别的顶级会议中发现了很多有趣的东 ...

  10. avalonJS入门

    前端神器avalonJS入门(一) posted @ 2014-10-31 17:44 vajoy 阅读(8759) 评论(42) 编辑 收藏   avalonJS是司徒正美开发和维护的前端mvvm框 ...