转自:http://www.cnblogs.com/wujd/archive/2012/08/17/2635309.html

listView中包含checkBox的时候,经常会发生其中的checkBox错乱的问题,大多时候的代码如下:

先看一下效果图:奇数行为选中状态,偶数行为非选中状态

具体代码:

布局文件:

 1 <?xml version="1.0" encoding="utf-8"?>
2 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
3 android:layout_width="fill_parent"
4 android:layout_height="fill_parent"
5 android:orientation="horizontal" >
6
7 <LinearLayout
8 android:id="@+id/layout"
9 android:layout_width="fill_parent"
10 android:layout_height="fill_parent">
11
12 <TextView
13 android:id="@+id/textView"
14 android:layout_width="wrap_content"
15 android:layout_height="wrap_content"/>
16
17 <CheckBox
18 android:id="@+id/checkBox"
19 android:layout_width="wrap_content"
20 android:layout_height="wrap_content"/>
21
22 </LinearLayout>
23
24 </LinearLayout>

JAVA CODE:

  1 package com.tony.ui.listview;
2
3 import java.util.ArrayList;
4 import java.util.List;
5
6 import com.tony.R;
7
8 import android.app.Activity;
9 import android.os.Bundle;
10 import android.view.LayoutInflater;
11 import android.view.View;
12 import android.view.ViewGroup;
13 import android.widget.BaseAdapter;
14 import android.widget.CheckBox;
15 import android.widget.CompoundButton;
16 import android.widget.CompoundButton.OnCheckedChangeListener;
17 import android.widget.LinearLayout;
18 import android.widget.ListView;
19 import android.widget.TextView;
20
21 public class ListViewCheckBox extends Activity{
22
23 private ListView listView;
24 private List<A> list;
25 private Adapter1 adapter1;
26
27 @Override
28 protected void onCreate(Bundle savedInstanceState) {
29 super.onCreate(savedInstanceState);
30 setContentView(R.layout.listview_checkbox);
31 initDate();
32 listView = (ListView)findViewById(R.id.listView);
33 adapter1 = new Adapter1();
34 listView.setAdapter(adapter1);
35 }
36
37
38 /**
39 * 模拟40个数据,奇数数据为选中状态,偶数数据为非选中状态
40 */
41 private void initDate(){
42 list = new ArrayList<A>();
43 A a;
44 for(int i=0;i<40;i++){
45 if(i%2==0){
46 a = new A(i+"号位",A.TYPE_NOCHECKED);
47 list.add(a);
48 }else{
49 a = new A(i+"号位",A.TYPE_CHECKED);
50 list.add(a);
51 }
52 }
53 }
54
55 class Adapter1 extends BaseAdapter{
56
57 @Override
58 public int getCount() {
59 return list.size();
60 }
61
62 @Override
63 public Object getItem(int position) {
64 return null;
65 }
66
67 @Override
68 public long getItemId(int position) {
69 return 0;
70 }
71
72 @Override
73 public View getView(int position, View convertView, ViewGroup parent) {
74 final int index = position;
75 ViewHolder viewHolder;
76 if(convertView == null){
77 viewHolder = new ViewHolder();
78 convertView = LayoutInflater.from(ListViewCheckBox.this).inflate(R.layout.listview_checkbox_item, null);
79 viewHolder.layout = (LinearLayout)convertView.findViewById(R.id.layout);
80 viewHolder.textView = (TextView)convertView.findViewById(R.id.textView);
81 viewHolder.checkBox = (CheckBox)convertView.findViewById(R.id.checkBox);
82 convertView.setTag(viewHolder);
83 }else{
84 viewHolder = (ViewHolder)convertView.getTag();
85 }
86
87
88 viewHolder.textView.setText(list.get(position).name);
89 if(list.get(position).type == A.TYPE_CHECKED){
90 viewHolder.checkBox.setChecked(true);
91 }else{
92 viewHolder.checkBox.setChecked(false);
93 }
94
95 /*点击checkBox所在行改变checkBox状态*/
96 /*final ViewHolder vv = viewHolder;
97 viewHolder.layout.setOnClickListener(new OnClickListener() {
98 @Override
99 public void onClick(View v) {
100 if(vv.checkBox.isChecked()){
101 vv.checkBox.setChecked(false);
102 list.get(index).type = TYPE_CHECKED;
103 }else{
104 vv.checkBox.setChecked(true);
105 list.get(index).type = TYPE_NOCHECKED;
106 }
107
108 }
109 });*/
110
111 viewHolder.checkBox.setOnCheckedChangeListener(new OnCheckedChangeListener() {
112 @Override
113 public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
114 if(isChecked){
115 list.get(index).type = A.TYPE_CHECKED;
116 }else{
117 list.get(index).type = A.TYPE_NOCHECKED;
118 }
119 }
120 });
121
122 return convertView;
123 }
124 }
125
126 class ViewHolder{
127 LinearLayout layout;
128 TextView textView;
129 CheckBox checkBox;
130 }
131
132 class A {
133
134 public static final int TYPE_CHECKED = 1;
135 public static final int TYPE_NOCHECKED = 0;
136
137 String name;
138 int type;
139
140 public A(String name,int type){
141 this.name = name;
142 this.type = type;
143 }
144 }
145 }

以上代码就是根据List集合中的对象的类型来设置checkBox是否为选中状态,当用户点击checkBox的时候,程序根据checkBox是否选中来将其状态保存至list集合对象中,相信很多人第一次做的时候会信心满满地认为这样的逻辑简直天衣无缝。但结果是,测试的时候还没有点击checkBox来改变其状态,只是简单地上下拉动listView的时候就会发现,好像事情没有想象的那么简单。

效果图如下:

只要视力不算太差的人便一眼可以看出,按照程序的逻辑怎么连续两个checkBox的状态会一样呢,恩,肯定是模拟器神经错乱了。

解决方案:

  很多人给出的两种解决办法

  1:上来就说是因为convertview对象共用的原因,不能用convetView,而是每次getView()的时候都new一个对象的view出来.这种办法大概是用屁股想出来的.

  2:即然错乱,那我就自己再弄一个集合保存checkBox的状态,再错乱,弄死你.即然adapter里有一个list集合里保存checkBox的状态了,为什么还要自己再保存一次checkBox的状态呢,不是多此一举吗?

  PS:提供这两种办法的人都没有解释到底是为什么错乱.下面来尝试分析一下:

  1:首先分析下viewHolder.checkBox.setOnCheckedChangeListener(new OnCheckedChangeListener()...);

     这句话,就是给checkBox添加一个监听器,如果checkBox的状态改变了,那么系统就会自动回调里面的onCheckedChange()方法.

而文中onCheckedChange()方法里写的是记录这次改变后checkBox状态的代码.

  2:再接着分析下if(list.get(position).type == A.TYPE_CHECKED){...},这部分代码是根据list集合里的对象属性初始化view里checkBox是否应该是选择状态.

  3:我上下滑动listView的时候,checkBox的状态就错乱了,根据第二点的分析,无论如何checkBox的状态都不会错乱,除非list集合里对象的属性已经被改变了,到底是什么地方改变了它?

  4:文中只有一个地方写了改变list集合里checkBox对象属性的地方,那就是第一点里提到的OnCheckedChangeListener()方法。它被执行了?

   怎么回事,不可能吧,打个断点,跑一下便知上下滑动listView的时候确实停了下来.

  5:这便是convertView的功能,因为不管listView里显示多少条数据,都只是共用那么几个对象,然后我们的代码每一次把得到的对象重新赋值而已。

   对了,正是在这赋值的时候出了问题,假设android系统给我们生成了10个共用view对象,第一个view对象在第一屏的时候需要显示成"未选择"状态,而到了第二屏的时候,却要显示成"选择"状态,但由于是共用的同一个对象,根据第一点得知当checkBox的状态改变的时候,会调用onCheckedChange()方法。

  6:也许有人会怀疑就算它调用了onCheckedChanged()方法,那又如何?onCheckedChange()方法里的代码还是将当前是否为选中状态保存到了list集合里,当我再次显示时还是会根据第二点里提到的代码来正确地显示,是的,代码会根据当前index来改变list集体的属性.关键就在这里,这个index真的是对的吗?测试一下便知:

    

    在onCheckedChange()方法里打印一下index, 当快速向下滑动的时候,index的值如下: 

  7:共用的对象有10(这里举例,并不是一定)个,当onCheckedChange()方法调用的时候,至少也是共用对象用光的时候,再从第一个共用对象用的时候才会打印,那时的index也应该是从10开始,为什么打印的结果里会出现0?

  8:这是由于代码的顺序决定的,根据上面的代码可以看出,添加监听器的代码在初始化checkBox属性的代码之后,也就是说当初始化checkBox属性时,由于可能改变其状态,导致调用了onCheckedChange()方法,而这个监听器是在上一次初始化的时候添加的,那么当然其index就是上一次的positon值,而不是本次的,所以每次保存checkBox属性状态的时候,都把值赋到的list集合里其它对象上去了,而不是与本次index相关的对象上,这才是发生莫名其妙错乱的真正原因.

  9:解决办法:由于是因为index错误造成的,那么只要保证index值与当前positon保持一至即可,只要把添加监听器的方法加到初始化view中checkBox状态的代码之前即可.这样即始由于初始化造成调用了onCheckedChange()方法,也因为其中index值是最新的,而依然不会错乱.

代码示例:

 1 class Adapter1 extends BaseAdapter{
2
3 @Override
4 public int getCount() {
5 return list.size();
6 }
7
8 @Override
9 public Object getItem(int position) {
10 return null;
11 }
12
13 @Override
14 public long getItemId(int position) {
15 return 0;
16 }
17
18 @Override
19 public View getView(int position, View convertView, ViewGroup parent) {
20 final int index = position;
21 ViewHolder viewHolder;
22 if(convertView == null){
23 viewHolder = new ViewHolder();
24 convertView = LayoutInflater.from(ListViewCheckBox.this).inflate(R.layout.listview_checkbox_item, null);
25 viewHolder.layout = (LinearLayout)convertView.findViewById(R.id.layout);
26 viewHolder.textView = (TextView)convertView.findViewById(R.id.textView);
27 viewHolder.checkBox = (CheckBox)convertView.findViewById(R.id.checkBox);
28 convertView.setTag(viewHolder);
29 }else{
30 viewHolder = (ViewHolder)convertView.getTag();
31 }
32
33 viewHolder.checkBox.setOnCheckedChangeListener(new OnCheckedChangeListener() {
34 @Override
35 public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
36 if(isChecked){
37 list.get(index).type = A.TYPE_CHECKED;
38 }else{
39 list.get(index).type = A.TYPE_NOCHECKED;
40 }
41 }
42 });
43
44 viewHolder.textView.setText(list.get(position).name);
45 if(list.get(position).type == A.TYPE_CHECKED){
46 viewHolder.checkBox.setChecked(true);
47 }else{
48 viewHolder.checkBox.setChecked(false);
49 }
50 return convertView;
51 }
52 }

总结:

  其实解决办法就一句话,"只要把添加监听器的方法加到初始化view中checkBox状态的代码之前即可. "

Android ListView CheckBox状态错乱(转)的更多相关文章

  1. RecyclerView item 状态错乱

    adapter中: private List<Integer> checkboxUserIdList = new ArrayList<>(); 在如下这个方法中: public ...

  2. [置顶] android ListView包含Checkbox滑动时状态改变

    题外话: 在xamarin android的开发中基本上所有人都会遇到这个小小的坎,的确有点麻烦,当时我也折腾了好一半天,如果你能看到这篇博客,说明你和我当初也是一样的焦灼,如果你想解决掉这个小小的坎 ...

  3. android 当ListView滚动时自动调用 onCheckedChanged 导致CheckBox 状态不停变化 的解决办法

    今天在做一个含有CheckBox 的ListView时,发现当初始化CheckBox的状态后, 滚动ListView,其中CheckBox 的选中状态不停的发生变化.最后发现原因是 ListView滚 ...

  4. android listview 使用checkbox问题

    在android中使用listview时需要了解listview加载数据的原理,为了避免listview由于列表项过多每次需要进行new造成性能低下的问题,android中的listview使用了控件 ...

  5. Android 带checkbox的listView 实现多选,全选,反选,删除

    activity_main.xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android&qu ...

  6. Android 带checkbox的listView 实现多选,全选,反选

    由于listview的一些特性,刚开始写这种需求的功能的时候都会碰到一些问题,重点就是存储每个checkbox的状态值,在这里分享出了完美解决方法:     布局文件: [html]   <?x ...

  7. 【转】Android 带checkbox的listView 实现多选,全选,反选 -- 不错

    原文网址:http://blog.csdn.net/onlyonecoder/article/details/8687811 Demo地址(0分资源):http://download.csdn.net ...

  8. 【转】Android 带checkbox的listView 实现多选,全选,反选----解决checkbox错位问题

    原文网址:http://blog.csdn.net/onlyonecoder/article/details/8687811 Demo地址(0分资源):http://download.csdn.net ...

  9. Android ListView 中的checkbox

    Q:ListView + CheckBox 当上下滚动的时候有事会自动选中或取消 A:这个与ListView的缓存机制有关.当你屏幕滚动后,ListView中的item选项视图先检查缓存中是否有视图, ...

随机推荐

  1. SQL记录-PLSQL事务

    PL/SQL事务   数据库事务是一个工作的原子单元,其可以由一个或多个相关的SQL语句组成.所谓的原子性就是数据库的修改所带来的构成事务的SQL语句可以集体被提交,即永久到数据库或从数据库中(撤消) ...

  2. bzoj千题计划218:bzoj2333: [SCOI2011]棘手的操作

    http://www.lydsy.com/JudgeOnline/problem.php?id=2333 上次那个是线段树,再发一个左偏树 维护两种左偏树 第一种是对每个联通块维护一个左偏树 第二种是 ...

  3. springMvc + Maven 项目提示 hessian 依赖包 无法下载;

    首先 从 https://github.com/alibaba/dubbo/archive/master.zip 下载最新的 dubbo 源码包到本地某个目录, 解压出来: cmd 进入该目录: 执行 ...

  4. JavaScript编写风格指南 (二)

    七:注释 // 频繁的使用注释有助于他人理解你的代码// 1.代码晦涩难懂// 2.可能被误认为是错误的代码// 3.必要但不明显的针对特定浏览器的代码// 4.对于对象,方法或者属性,生成文档是有必 ...

  5. Guava BiMap

    BiMap主要的就是用于key,value的互相映射,获取相互的结果,还保证值value是唯一的,key相同覆盖原来值. 举例: BiMap<Integer, String> empIDN ...

  6. 【转】如何评价 Apple 新推出的编程语言 Swift?

    如何评价 Apple 新推出的编程语言 Swift? 原文地址:http://www.zhihu.com/question/24002984 评价:如果你会Objective-C,你不需要去看它.   ...

  7. aps.net webform框架下页面服务器端控件和html控件用法

    (1)select 下拉框 前端: <select name="gameserverlist" id="gameserverlist" runat=&qu ...

  8. ZYNQ. Interrupt(2)SPI.AXI TIMER

    Shared Peripheral Interrupts (SPI) SPI 可以接收来自PL的中断,这里使用PL模块 AXI Timer 的中断模式,并连接到CPU. AXI TIMER 定时器,内 ...

  9. 搭建RabbitMQ集群(通用)

    RabbitMQ在Erlang node(节点)上 Erlang天生具有集群特性,非常好搭建集群,每一个节点(node)上具有一个叫erlang.Cookie的东西,也是一个标识符,可以互认. 1). ...

  10. 联通-长春处,FDD和TDD宏站,数据业务接入时延期望值默认值应为80ms

    有小坑 备注:若已经跑过V5.40.00_Alpha1_Baseline.sql或V5.30.02_Beta_TO_V5.40.00_Alpha1.sql的脚本,再次运行升级脚本修改不成功,需手动在数 ...