[转载]解决Android在listview中checkbox批量操作问题

在Android某些开 发需求当中,有时候需要在listveiw中加入checkbox实现单选,多选操作。表面上看上去只是改变checkbox那么简单,然而实际开发中, 实现起来并不是那么得心应手。尤其当listview比较多(比如屏幕最多只能显示10个item,但总共有12个item,也就是说listview的 item数大于屏幕能够显示的item数)滑动屏幕的时候,由于适配器中getview()会重复使用被移除屏幕的item,所以会造成checkbox 选择状态不正常的现象。自己在开发中碰到这样的问题很是苦恼,查了下资料,发现网上很少没有针对这类批量操作并没有一个完整的例子。搜了很多篇帖子才完美 的实现这一常用的操作。所以在这里把这个Demo贴出来,供大家参考,希望能对大家有所帮助。

      主界面的布局main.xml    这个就不多说什么

 

  1. <LinearLayout  
  2.         android:orientation="vertical"  
  3.         android:layout_width="fill_parent"  
  4.         android:layout_height="wrap_content"  
  5.          >  
  6.     <TextView   
  7.         android:layout_width="fill_parent"  
  8.         android:layout_height="100dip"  
  9.         android:text="原创:Simtice                                  QQ:512375320"  
  10.         android:layout_marginLeft="10dip"  
  11.         />  
  12.     </LinearLayout><?xml version="1.0" encoding="utf-8"?>  
  13. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  14.     android:layout_width="fill_parent"  
  15.     android:layout_height="wrap_content"  
  16.     android:orientation="vertical" >  
  17.     <LinearLayout  
  18.         android:orientation="vertical"  
  19.         android:layout_width="fill_parent"  
  20.         android:layout_height="wrap_content"  
  21.          >  
  22.         <TextView   
  23.             android:id="@+id/tv"  
  24.             android:layout_width="fill_parent"  
  25.             android:layout_height="50dip"  
  26.             android:textColor="#FCFCFC"  
  27.             android:textSize="11pt"  
  28.             android:gravity="center_vertical"  
  29.             android:layout_marginLeft="10dip"  
  30.             />   
  31.     <ListView  
  32.         android:id="@+id/lv"  
  33.         android:layout_width="fill_parent"  
  34.         android:layout_height="381dip"  
  35.         android:cacheColorHint ="#00000000"  
  36.          ></ListView>  
  37.     </LinearLayout>  
  38.     <RelativeLayout   
  39.         android:layout_width="fill_parent"  
  40.         android:layout_height="53dip"  
  41.         android:orientation="horizontal"  
  42.         >  
  43.         <Button   
  44.             android:id="@+id/selectall"  
  45.             android:layout_width="80dip"      
  46.             android:layout_height="50dip"  
  47.             android:layout_marginLeft="20dip"  
  48.             android:text="全选"  
  49.             android:gravity="center"  
  50.             />  
  51.         <Button   
  52.             android:id="@+id/inverseselect"  
  53.             android:layout_width="80dip"      
  54.             android:layout_height="50dip"  
  55.             android:layout_marginLeft="118dip"  
  56.             android:text="反选"  
  57.             android:gravity="center"  
  58.             />  
  59.         <Button   
  60.             android:id="@+id/cancel"  
  61.             android:layout_width="80dip"      
  62.             android:layout_height="50dip"  
  63.             android:layout_marginLeft="213dip"  
  64.             android:text="取消已选"  
  65.             android:gravity="center"  
  66.             />  
  67.     </RelativeLayout>  
  68.            <LinearLayout  
  69.                 android:orientation="vertical"  
  70.                 android:layout_width="fill_parent"  
  71.                 android:layout_height="wrap_content"  
  72.                >  
  73.              <TextView   
  74.                  android:layout_width="fill_parent"  
  75.                  android:layout_height="100dip"  
  76.                  android:text="原创:Simtice                                  QQ:512375320"  
  77.                  android:layout_marginLeft="10dip"  
  78.                />  
  79.          </LinearLayout>  
  80.  </LinearLayout>  


ListView每个item的布局,listviewitem.xml:

这里需要注意的是,由于checkbox的点击事件优先级比listview的高,所以要添加android:focusable="false"属性,使得checkbox初始的时候没有获取焦点。

另外这里是点击ListView的item控制checkbox的状态改变,也就是让item接收clik事件,所以需要加上android:focusableInTouchMode="false"这一属性。

  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <RelativeLayout  xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:layout_width="fill_parent"  
  4.     android:layout_height="55dip"  
  5.     android:orientation="horizontal"  
  6.     android:layout_marginTop="20dip"  
  7.     >   
  8.     <TextView   
  9.         android:id="@+id/item_tv"  
  10.         android:layout_width="267dip"  
  11.         android:layout_height="40dip"  
  12.         android:textSize="10pt"  
  13.         android:gravity="center_vertical"  
  14.         android:layout_marginLeft="10dip"  
  15.         />  
  16.     <CheckBox   
  17.         android:id="@+id/item_cb"  
  18.         android:layout_width="wrap_content"  
  19.         android:layout_height="wrap_content"  
  20.         android:focusable="false"      
  21.         android:focusableInTouchMode="false"      
  22.         android:clickable="false"    
  23.         android:layout_toRightOf="@id/item_tv"     
  24.         android:layout_alignParentTop="true"  
  25.         android:layout_marginRight="5dip"  
  26.          
  27.         />  
  28. </RelativeLayout <span style="font-size:16px;"></span>  

ViewHolder类

  1. package simtice.test.listview.viewholder;  
  2. import android.widget.CheckBox;  
  3. import android.widget.TextView;  
  4.   
  5. public class ViewHolder {  
  6.     public TextView tv = null;  
  7.     public CheckBox cb = null;  
  8. }  


为listview自定义适配器,该类为主Activity类MainActivity.java的内部类

 

  1. class MyAdapter extends BaseAdapter {  
  2.         @Override  
  3.         public int getCount() {  
  4.             return list.size();  
  5.         }  
  6.   
  7.         @Override  
  8.         public Item getItem(int arg0) {  
  9.             return list.get(arg0);  
  10.         }  
  11.   
  12.         @Override  
  13.         public long getItemId(int arg0) {  
  14.             return arg0;  
  15.         }  
  16.   
  17.         @Override  
  18.         public View getView(int position, View view, ViewGroup arg2) {  
  19.             System.out.println("getView " + position + " " + view);  
  20.             ViewHolder holder;  
  21.             if (view == null || (holder = (ViewHolder) view.getTag()) == null) {  
  22.                 view = View.inflate(MainActivity.this, R.layout.listviewitem,  
  23.                         null);  
  24.                 holder = new ViewHolder();  
  25.                 holder.tv = (TextView) view.findViewById(R.id.item_tv);  
  26.                 holder.cb = (CheckBox) view.findViewById(R.id.item_cb);  
  27.                 view.setTag(holder);  
  28.             }  
  29.             Item item = getItem(position);  
  30.             holder.tv.setText(item.name);  
  31.             holder.cb.setChecked(item.status);  
  32.             return view;  
  33.         }  


最后,最重要的就是MainActivity.java中一些事件响应的处理

  1. public class MainActivity extends Activity implements OnClickListener {  
  2.     TextView tv = null;  
  3.     ListView lv = null;  
  4.     String name[] = { "G1", "G2", "G3", "G4", "G5", "G6", "G7", "G8", "G9",  
  5.             "G10", "G11", "G12", "G13", "G14" };  
  6.   
  7.     private List<Item> list;  
  8.     private List<String> data;  
  9.     private MyAdapter adapter;  
  10.   
  11.     @Override  
  12.     public void onCreate(Bundle savedInstanceState) {  
  13.         super.onCreate(savedInstanceState);  
  14.         setContentView(R.layout.main);  
  15.         tv = (TextView) this.findViewById(R.id.tv);  
  16.         lv = (ListView) this.findViewById(R.id.lv);  
  17.   
  18.         this.findViewById(R.id.selectall).setOnClickListener(this);  
  19.         this.findViewById(R.id.inverseselect).setOnClickListener(this);  
  20.         this.findViewById(R.id.cancel).setOnClickListener(this);  
  21.   
  22.         lv.setOnItemClickListener(new OnItemClickListener() {  
  23.             @Override  
  24.             public void onItemClick(AdapterView<?> arg0, View arg1, int arg2,  
  25.                     long arg3) {  
  26.                 Item item = list.get(arg2);  
  27.                 item.status = !item.status;// 取反  
  28.                 initAdapter();  
  29.             }  
  30.         });  
  31.   
  32.         init();  
  33.   
  34.     }  
  35.   
  36.     @Override  
  37.     public void onClick(View v) {  
  38.         switch (v.getId()) {  
  39.         case R.id.selectall:  
  40.             int size1 = list.size();  
  41.             for (int i = 0; i < size1; i++) {  
  42.                 list.get(i).status = true;  
  43.             }  
  44.             break;  
  45.         case R.id.inverseselect:  
  46.             int size2 = list.size();  
  47.             for (int i = 0; i < size2; i++) {  
  48.                 Item item = list.get(i);  
  49.                 item.status = !item.status;// 取反  
  50.             }  
  51.             break;  
  52.         case R.id.cancel:  
  53.             int size3 = list.size();  
  54.             for (int i = 0; i < size3; i++) {  
  55.                 list.get(i).status = false;  
  56.             }  
  57.             break;  
  58.         }  
  59.         initAdapter();  
  60.     }  
  61.   
  62.     // 数据初始化  
  63.     private void init() {  
  64.         if (list == null)  
  65.             list = new ArrayList<Item>();  
  66.         else  
  67.             list.clear();  
  68.         if (data == null)  
  69.             data = new ArrayList<String>();  
  70.         for (String s : name) {  
  71.             list.add(new Item(s, false));  
  72.         }  
  73.         initAdapter();  
  74.     }  
  75.   
  76.     // 刷新适配器  
  77.     public void initAdapter() {  
  78.         if (adapter == null) {  
  79.             adapter = new MyAdapter();  
  80.             lv.setAdapter(adapter);  
  81.         } else {  
  82.             adapter.notifyDataSetChanged();  
  83.         }  
  84.   
  85.         int size = list.size();  
  86.         data.clear();  
  87.         for (int i = 0; i < size; i++) {  
  88.             if (list.get(i).status)  
  89.                  /*这里可以处理checkbox为true时的事件*/    
  90.                                data.add(name[i]);    
  91.             else                    
  92.                /*为false时的事件*/    
  93.                 data.remove(name[i]);  
  94.         }  
  95.         tv.setText("已选中 " + data.size() + " 项");  
  96.     }  
  97.   
  98.     // 为listview自定义适配器内部类  
  99.     class MyAdapter extends BaseAdapter {  
  100.         ...  
  101.     }  
  102.   
  103.     class Item {  
  104.         public String name;  
  105.         public boolean status = false;  
  106.   
  107.         public Item(String name, boolean b) {  
  108.             this.name = name;  
  109.             this.status = b;  
  110.         }  
  111.     }  

我选择了G2、G3、G11三项,现在屏幕滑动到底部,看以看到状态保存的很好,TextView显示已选中3项。全选、反选、取消已选功能正常,多选操作完美解决!

posted @ 2015-01-14 22:39  chenlinyunyi  阅读(232)  评论(0编辑  收藏  举报