带有 CheckBox 的 ListView 的奇怪行为

Posted

技术标签:

【中文标题】带有 CheckBox 的 ListView 的奇怪行为【英文标题】:Weird Behaviour of ListView with CheckBox 【发布时间】:2012-01-10 13:00:53 【问题描述】:

我在 ListView 上方有一个 CheckBox,其中包含 Checkbox 项。因此,当我选中复选框时,我会应用 notifyDataSetChanged() 来检查列表中的所有项目。所以当时我两次获取 getView 方法的日志,这意味着当我只调用一次 notifyDataSetChanged() 时,getView 被调用了两次。那么,谁能告诉我为什么我会两次获取日志?还有为什么 getView() 会被调用两次?

主类 -

checkAll = (CheckBox) findViewById(R.id.my_check_box);
        checkAll.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() 

            @Override
            public void onCheckedChanged(CompoundButton checkbox, boolean arg1) 
                adapter.notifyDataSetChanged();
            
        );

适配器类的代码 -

    public class myAdapter extends ArrayAdapter<Model> 

    private final List<Model> list;
    private final Activity context;
    private CheckBox checkAll;

    public myAdapter(Activity context, List<Model> list, CheckBox checkAll) 
        super(context, R.layout.row, list);
        this.context = context;
        this.list = list;
        this.checkAll = checkAll;

    

    static class ViewHolder 
        protected TextView text;
        protected CheckBox checkbox;
    

    @Override
    public View getView(final int position, View convertView, ViewGroup parent) 
        View view = null;
        if (convertView == null) 
            LayoutInflater inflator = context.getLayoutInflater();
            view = inflator.inflate(R.layout.row, null);
            final ViewHolder viewHolder = new ViewHolder();
            viewHolder.text = (TextView) view.findViewById(R.id.label);
            viewHolder.checkbox = (CheckBox) view.findViewById(R.id.check);
            viewHolder.checkbox
                    .setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() 

                        @Override
                        public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) 
                            Model element = (Model) viewHolder.checkbox.getTag();
                            element.setSelected(buttonView.isChecked());
                        
                    );
            view.setTag(viewHolder);
            viewHolder.checkbox.setTag(list.get(position));
         else 
            view = convertView;
            ((ViewHolder) view.getTag()).checkbox.setTag(list.get(position));
        
        ViewHolder holder = (ViewHolder) view.getTag();
        holder.text.setText(list.get(position).getName());


        if(checkAll.isChecked() == true)
            System.out.println(position+" checked th position");
        
        else if(checkAll.isChecked() == false)
            System.out.println(position+" not checked th position");
        

        holder.checkbox.setChecked(list.get(position).isSelected());    
        return view;
    

LogCat 输出当我可以 adapter.notifyDataSetChanged(); 在 CheckAll 复选框上检查更改侦听器时。

11-30 20:31:33.751: INFO/System.out(1300): 0 checked th position
11-30 20:31:33.761: INFO/System.out(1300): 1 checked th position
11-30 20:31:33.770: INFO/System.out(1300): 2 checked th position
11-30 20:31:33.780: INFO/System.out(1300): 3 checked th position
11-30 20:31:33.810: INFO/System.out(1300): 4 checked th position
11-30 20:31:33.810: INFO/System.out(1300): 0 checked th position
11-30 20:31:33.831: INFO/System.out(1300): 1 checked th position
11-30 20:31:33.831: INFO/System.out(1300): 2 checked th position
11-30 20:31:33.851: INFO/System.out(1300): 3 checked th position
11-30 20:31:33.860: INFO/System.out(1300): 4 checked th position

您可以看到我的 Logcat 输出两次得到相同的输出。那么有人能说出为什么会这样吗?

【问题讨论】:

这个问题也和我一起提出... +1 ***.com/questions/2618272/… 看看这个 @Lalit Poptani 你滚动吗? @Venky 不,我没有滚动,我只有 4 个项目。 @LalitPoptani 你好,我正在滚动和多选,然后全选,单选。 Single , multiple ,选择所有工作正常但滚动时位置发生变化,并且全部取消选择。那么我该如何解决这个问题。 【参考方案1】:

尝试让您的ListView 高度为fill_parent。 Lisiview 尝试自己根据提供的空间进行一些调整,这是我学到的,看起来这就是背后的原因。

看看有没有帮助。

【讨论】:

这意味着当你想将你的列表视图设置在屏幕的一半时..你无法解决上述问题? 如果你想将你的列表视图设置在屏幕的一半,你可以使用 android:layout_weight 【参考方案2】:

这并不是你问题的真正答案——只是当我看到这个时,我的眼睛有点疼:

if(checkAll.isChecked() == true)
    System.out.println(position+" checked th position");

else if(checkAll.isChecked() == false)
    System.out.println(position+" not checked th position");

if-statement 需要 boolean 表达式,并且由于方法 isChecked() 返回 boolean,因此无需将其与 true/false 匹配。此外,由于 boolean 只能是 truefalse,因此无需同时检查两者。简而言之,您可以将上述内容归结为:

if( checkAll.isChecked() )
  System.out.println(position+" checked th position");
else
  System.out.println(position+" not checked th position");

哪个更漂亮:-)

不要将此视为对您的编程技能的批评 - 只需将其视为学习机会。

【讨论】:

嗯,它只是一个演示应用程序,用于准确了解我的问题。也谢谢。 太搞笑了,kaspermoerch!试试这个:System.out.println(position + (checkAll.isChecked() ? " checked th position" : " not checked th position")); 这也行得通,杰夫。我确实发现,应该谨慎使用三元运算符。它并不总能提高可读性。【参考方案3】:

这是因为 checkchangelistener 中的 viewHolder 会随着列表滚动而改变。因此使用这种方式无法实现第 i 个复选框和 viewHolder 之间的映射。

创建单独的类 checkchangelistener 实现 checnChecnglistener 。通过构造函数传递 checkchangelistener。

EG。在 getView 之外

class MyCheckChangeLsitenet implements CompoundButton.OnCheckedChangeListener

ViewHolder viewHolder
 MyCheckChangeLsitenet(ViewHolder viewHolder)
 
this.viewHolder =  viewHolder ;

                        @Override
                        public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) 
                            Model element = (Model) viewHolder.checkbox.getTag();
                            element.setSelected(buttonView.isChecked());
                        
                    );

在 GetView 中

viewHolder.checkbox .setOnCheckedChangeListener(new MycheckChangeListenet(viewHolder))

【讨论】:

我目前没有滚动列表,虽然它给出了同样的问题。 忘记滚动这个词。试图让您熟悉内部 listView 处理。通过描述的方式。它会解决你的问题 你是对的@ShailendraSinghRajawat...但是通用模型不是这样的,还有其他简单的方法吗?

以上是关于带有 CheckBox 的 ListView 的奇怪行为的主要内容,如果未能解决你的问题,请参考以下文章

带有复选框的类似 Gmail 的 ListView(并使用 ActionBar)

为啥当我在选中 CheckBox 的情况下滚动 ListView 时,它会变为未选中状态?

带有可点击/可编辑小部件的 ListView

ListView 中的自定义复选框

ListView中CheckBox使用问题

C# winform ListView+CheckBox的做法