RecyclerView的item有EditText时复用问题,及RecyclerView.Adapter的onBindViewHolder参数position不能final。

RecyclerView已经基本替代了ListView和GridView成为开发中使用最多的控件,在使用的过程中遇到的问题在此记录一下。
1,RecyclerView的item中有EditText时,当你在某个item的EditText输入内容之后滑动的话,由于复用就会导致你输入的内容出现的复用的新item中,再滑回去可能你刚才输入的内容就没有了,这个肯定不行的。
解决办法:a,最简单的是把EditText换成TextView,写一个单独的输入功能,例如一个能输入的Dialog,每次点击TextView弹窗输入Dialog即可。b,定义一个数组存放每个item输入的内容,这样也可以保证顺序不乱。
2,在RecyclerView.Adapter的onBindViewHolder方法中,假如有监听器之类的东西里用到了局部变量是需要把这些局部变量定义为final,这里如果用到position,我们通常不能把position设置为final,因为这样可能会出问题,比如你的item中有CheckBox之类的控件,就会导致选中状态出现问题,这个可以想办法解决,但是不是问题的根本所在,我们通常可以用holder.getAdapterPosition()来代替position,这样就不会出现问题了。

结合上面问题,代码如下:

public class ListAdapter extends RecyclerView.Adapter<ListAdapter.ListHolder> {

    private Context context;
    private ArrayList<ListBean> listBeans;
    private String[] inputs;

    public ListAdapter(Context context, ArrayList<ListBean> listBeans) {
        this.context = context;
        this.listBeans = listBeans;
        inputs = new String[listBeans.size()];
    }

    @NonNull
    @Override
    public ListHolder onCreateViewHolder(@NonNull ViewGroup parent, int position) {
        return new ListHolder(LayoutInflater.from(context).inflate(R.layout.item_recyclerview, parent, false));
    }

    @Override
    public void onBindViewHolder(@NonNull final ListHolder holder, int position) {
        final ListBean listBean = listBeans.get(position);
        if (listBean != null) {
            holder.tvTitle.setText(listBean.title);
            holder.checkbox.setChecked(listBean.isChecked);
            holder.etInput.setText(inputs[holder.getAdapterPosition()]);
            holder.etInput.addTextChangedListener(new TextWatcher() {
                @Override
                public void beforeTextChanged(CharSequence s, int start, int count, int after) {

                }

                @Override
                public void onTextChanged(CharSequence s, int start, int before, int count) {
                    inputs[holder.getAdapterPosition()] = s.toString();
                }

                @Override
                public void afterTextChanged(Editable s) {

                }
            });
            holder.checkbox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
                @Override
                public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
                    listBeans.get(holder.getAdapterPosition()).isChecked = isChecked;
                }
            });
        }
    }

    @Override
    public int getItemCount() {
        return listBeans == null ? 0 : listBeans.size();
    }

    class ListHolder extends RecyclerView.ViewHolder {

        @BindView(R.id.tvTitle)
        TextView tvTitle;
        @BindView(R.id.etInput)
        EditText etInput;
        @BindView(R.id.checkbox)
        CheckBox checkbox;

        public ListHolder(@NonNull View itemView) {
            super(itemView);
            ButterKnife.bind(this, itemView);
        }
    }
### Android ListView Adapter 的使用方法 在 Android 开发中,`ListView` 是一种常见的 UI 控件,用于展示列表形式的数据。为了使 `ListView` 能够动态加载并显示数据,通常需要为其设置一个自定义的适配器(Adapter)。以下是关于如何为 `ListView` 设置适配器以及常见问题的解决方案。 #### 1. 创建自定义 BaseAdapter 当默认的 `ArrayAdapter` 和其他内置适配器无法满足需求,可以创建一个继承自 `BaseAdapter` 的类来实现更复杂的功能。下面是一个简单的例子: ```java public class CustomAdapter extends BaseAdapter { private Context context; private List<String> dataList; public CustomAdapter(Context context, List<String> dataList) { this.context = context; this.dataList = dataList; } @Override public int getCount() { return dataList.size(); } @Override public Object getItem(int position) { return dataList.get(position); } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { ViewHolder viewHolder; if (convertView == null) { viewHolder = new ViewHolder(); LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); convertView = inflater.inflate(R.layout.list_item_layout, null); viewHolder.textView = convertView.findViewById(R.id.text_view); convertView.setTag(viewHolder); } else { viewHolder = (ViewHolder) convertView.getTag(); } viewHolder.textView.setText(dataList.get(position)); return convertView; } static class ViewHolder { TextView textView; } } ``` 此代码展示了如何通过 `BaseAdapter` 实现自定义视图逻辑[^1]。 --- #### 2. 将适配器绑定到 ListView 一旦实现了自定义适配器,就可以将其绑定到 `ListView` 上: ```java ListView listView = findViewById(R.id.listView); CustomAdapter adapter = new CustomAdapter(this, yourDataList); listView.setAdapter(adapter); ``` 这一步骤确保了 `ListView` 可以正确地获取和渲染数据。 --- #### 3. 常见问题及其解决方案 ##### **问题 1**: `getView()` 方法未被调用 如果发现 `getView()` 方法从未被执行,则可能是由于以下几个原因: - 数据源为空 (`data.size() == 0`)。 - `ListView` 的高度或其父布局的高度可能设置了不合适的值(如 `wrap_content`),导致系统认为无需绘制任何项。 - 同一 `ListView` 对象绑定了多个不同的适配器实例。 - 数据源对象发生了变化(例如重新初始化了一个新的 `ArrayList`)[^3]。 **解决办法**: - 确认数据源非空。 - 修改 `ListView` 或其父容器的高度为固定值或者 `match_parent`。 - 确保每次更新数据都调用了 `notifyDataSetChanged()` 并保持同一数据源引用。 --- ##### **问题 2**: 列表中的编辑框(`EditText`)数据重复刷新 当 `ListView` 中包含可交互控件(如 `EditText`),在滚动过程中可能会发生数据错乱的现象。这是因为 `ListView` 复用了旧的视图而没有及保存状态[^5]。 **解决办法**: 可以通过维护额外的状态变量记录每个位置上的输入内容,并在 `getView()` 方法中恢复这些状态。例如: ```java @Override public View getView(final int position, View convertView, ViewGroup parent) { ViewHolder viewHolder; if (convertView == null) { viewHolder = new ViewHolder(); LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); convertView = inflater.inflate(R.layout.list_item_with_edittext, null); viewHolder.editText = convertView.findViewById(R.id.edit_text); convertView.setTag(viewHolder); } else { viewHolder = (ViewHolder) convertView.getTag(); } String savedValue = editTextValues.get(position); // 获取之前存储的内容 if (savedValue != null && !savedValue.isEmpty()) { viewHolder.editText.setText(savedValue); } viewHolder.editText.addTextChangedListener(new TextWatcher() { @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) {} @Override public void onTextChanged(CharSequence s, int start, int before, int count) { editTextValues.put(position, s.toString()); // 动态保存当前输入 } @Override public void afterTextChanged(Editable s) {} }); return convertView; } static class ViewHolder { EditText editText; } ``` 上述代码片段解决了因视图复用而导致的 `EditText` 数据丢失问题。 --- ##### **问题 3**: 如何处理点击事件? 对于 `ListView` 来说,监听每一项的点击操作是非常重要的功能之一。可以在 XML 文件中指定 `onItemClick` 属性,也可以直接在 Java/Kotlin 代码中注册监听器: ```java listView.setOnItemClickListener((parent, view, position, id) -> { Toast.makeText(context, "Clicked item at position: " + position, Toast.LENGTH_SHORT).show(); }); ``` 此外,还可以通过重写 `getView()` 方法为特定子控件添加单独的点击回调函数[^4]。 --- ### 总结 以上介绍了如何为 `ListView` 设置适配器的方法,同也列举了一些开发过程中的典型问题及对应的解决方案。合理利用 `BaseAdapter` 提供的强大灵活性能够帮助开发者构建更加复杂的界面效果。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值