RecycleView 插入item或删除item后导致后面item中EditText数据错误的一种解决办法

众做周知,提高RecycleView性能的办法就是减少item刷新项,所以notifyItemRemoved,notifyItemRangeChangged结合使用是程序猿必须要学会使用的方法.然而如果你的item里面有EditText项时,麻烦就来了,当你调用notifyItemRangeChangded时,导致后续的item被重新加载,如果没有做好绑定数据的操作,那么恭喜你,EditText中的内容没了!就如我一样_:
在这里插入图片描述
我的业务需求就是当用户点击删除按钮后,当前栏目会被删除掉,只有当用户点击√按钮后数据才会被保存,
我发现如果用户如上图提前输入好数据后,点击删除掉其中的某一行后,后面的输入好的数据全部消失.当用户在点击添加按钮时,新加入的item理应没有数据,但奇怪的是EditText中有出现了用户在别的栏目输好的数据!
如下图,我删除了""22""那行数据后,后面item中的EditText中输好的数据也消失了:!
在这里插入图片描述
当我再添加一栏时,不应该出现的数据出现了!:在这里插入图片描述
要解决问题就必须明白问题出在哪:recycleview的复用导致的这种情况,调用notifyItemRangeChanged(position,itemcount)方法后,后面的item中的数据全部重新被bind新holder了,之前holder没有保存的数据清空.但之前item的bind好的holder并没有销毁,导致其中的EditText数据没有清空,在添加新item时复用之前的holder,所以之前的EditText中的数据有出现了.
解决办法就是使用setTag加payloads结合使用:
我的思路就是如果调用notifyItemRangeChanged使用(position,itemcount,payloads)函数,其中payloads可以是任意数值,主要起个提示作用;这个函数调用后,系统会调用onBindViewHolder(ViewHolder holder, int position, @NonNull List payloads)这个函数,其中holder中的数据视图都是是之前的,所以其中的view的数据都是之前的,position是recycleview已经重新定位好的,这时我们就可以使用
@Override public void onBindViewHolder(@NonNull ViewHolder holder, int position, @NonNull List<Object> payloads) { onBindViewHolder(holder, position); }
重新绑定视图.

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
如果您想要在 RecycleView 实现 ItemMove 操作并将 Item 插入到整体 Item 之后移动,可以尝试以下步骤: 1. 实现 ItemTouchHelper.Callback 接口,在其重写 onMove 和 onSwiped 方法,分别处理 Item 移动和删除操作。 2. 在 onMove 方法,获取被拖拽的 Item 的位置和目标位置,然后将该 Item 从原位置移除,再将其插入到目标位置。 3. 在插入 Item 之后,您需要重新计算整个列表所有 Item 的位置,确保它们按照正确的顺序排列。 下面是一个示例代码,供参考: ```java public class ItemMoveCallback extends ItemTouchHelper.Callback { private final ItemTouchHelperAdapter mAdapter; public ItemMoveCallback(ItemTouchHelperAdapter adapter) { mAdapter = adapter; } @Override public boolean isLongPressDragEnabled() { return true; } @Override public boolean isItemViewSwipeEnabled() { return false; } @Override public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) { int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN; int swipeFlags = 0; return makeMovementFlags(dragFlags, swipeFlags); } @Override public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) { int fromPosition = viewHolder.getAdapterPosition(); int toPosition = target.getAdapterPosition(); mAdapter.onItemMove(fromPosition, toPosition); return true; } @Override public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) { // do nothing } @Override public void onSelectedChanged(RecyclerView.ViewHolder viewHolder, int actionState) { if (actionState != ItemTouchHelper.ACTION_STATE_IDLE) { if (viewHolder instanceof ItemTouchHelperViewHolder) { ItemTouchHelperViewHolder itemViewHolder = (ItemTouchHelperViewHolder) viewHolder; itemViewHolder.onItemSelected(); } } super.onSelectedChanged(viewHolder, actionState); } @Override public void clearView(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) { super.clearView(recyclerView, viewHolder); if (viewHolder instanceof ItemTouchHelperViewHolder) { ItemTouchHelperViewHolder itemViewHolder = (ItemTouchHelperViewHolder) viewHolder; itemViewHolder.onItemClear(); } } } ``` 在 Adapter 实现 ItemTouchHelperAdapter 接口: ```java public interface ItemTouchHelperAdapter { void onItemMove(int fromPosition, int toPosition); } ``` 然后在 Adapter 实现 onItemMove 方法: ```java @Override public void onItemMove(int fromPosition, int toPosition) { // 将 fromPosition 对应的 Item 移除 MyItem item = mItems.remove(fromPosition); // 将 Item 插入到 toPosition 之后 mItems.add(toPosition + 1, item); // 重新计算所有 Item 的位置 notifyItemMoved(fromPosition, toPosition + 1); for (int i = 0; i < getItemCount(); i++) { MyItem item = mItems.get(i); item.setPosition(i); } } ``` 在上述示例代码,当用户将一个 Item 拖拽到另一个位置时,会调用 onMove 方法,并在该方法将该 Item 移动到目标位置之后。然后,我们需要调用 notifyItemMoved 方法通知 Adapter 该 Item 的位置已经发生了变化,并重新计算所有 Item 的位置。最后,我们需要更新每个 Item 的位置属性,以确保它们按照正确的顺序排列。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值