RecyclerView里notifyItemRemoved的坑

本文探讨了在使用RecyclerView时如何正确实现数据删除操作及动画效果。详细解释了为何直接在ViewHolder绑定方法中移除数据会导致索引错乱,并提供了正确的解决方案。包括使用getAdapterPosition获取当前有效的位置并结合notifyItemRemoved和notifyItemRangeChanged通知更新。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

RecyclerView很多时候是展示静态的数据,并不会有删除的操作,讲到RecyclerView时,会提到它提供了一个很好的展现删除操作动画的函数,代码片段一般是这样的

   @Override
        public void onBindViewHolder(final CommonViewHolder holder, final int position) {   
            holder.itemView.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    contentList.remove(position);
                    notifyItemRemoved(position);            
                }
            });
        }

这样写的话,很快会产生数据删除错乱和超出索引异常导致崩溃。
原因是函数里面的传入的参数position,它是在进行onBind操作时确定的,在删除单项后,已经出现在画面里的项不会再有调用onBind机会,这样它保留的position一直是未进行删除操作前的postion值。
对于尚未进入画面的单项来说,它会使用新的position值,这个值是正确的,如果在单项里加上下面的代码

  holder.textView.setText(contentList.get(position) + "#" + String.valueOf(position));

在删除第一屏的一项后,向上滚动,会发现新滚上来的一行和它上面的一行的textview显示是一样的。

解决办法也很简单

  //或者使用getLayoutPosition
  contentList.remove(holder.getAdapterPosition());
  notifyItemRemoved(holder.getAdapterPosition());

PS:推荐评论里提供的方法,谢谢
dodouaj
2016-03-30 16:59发表 [回复]
更好的解决办法是。
先remove,再notifyItemRemoved, 最后再notifyItemRangeChanged
remove:把数据从list中remove掉,
notifyItemRemoved:显示动画效果
notifyItemRangeChanged:对于被删掉的位置及其后range大小范围内的view进行重新onBindViewHolder

### 自定义 Android RecyclerView `notifyItemRemoved` 动画 为了实现自定义的 `RecyclerView` 删除项动画,可以创建并配置一个继承自 `DefaultItemAnimator` 的类来覆盖默认行为。通过这种方式能够完全控制删除操作期间发生的视觉变化。 #### 创建自定义 Item Animator 类 ```java public class CustomRemoveAnimator extends DefaultItemAnimator { @Override public boolean animateRemove(final ViewHolder holder) { final View view = holder.itemView; // 开启一个属性动画 final ViewPropertyAnimatorCompat animation = ViewCompat.animate(view); animation.setDuration(getRemoveDuration()) .alpha(0) .scaleX(0.5f).scaleY(0.5f) .setListener(new ViewPropertyAnimatorListenerAdapter() { @Override public void onAnimationStart(View view) { dispatchAnimationsStarted(); } @Override public void onAnimationEnd(View view) { animation.setListener(null); view.setVisibility(View.GONE); dispatchRemoveFinished(holder); dispatchAnimationsFinished(); } }).start(); return true; // 返回true表示有正在运行的动画 } } ``` 此代码片段展示了如何设置缩放和平滑淡出的效果作为移除项目的过渡效果[^1]。 #### 设置自定义 Animator 到 RecyclerView 为了让上述自定义动画生效,在初始化 `RecyclerView` 后需指定其使用的 `ItemAnimator`: ```java recyclerView.setItemAnimator(new CustomRemoveAnimator()); ``` 当调用 `adapter.notifyItemRemoved(int position)` 方法时,将会触发之前定义好的自定义动画逻辑而不是标准的行为[^2]。
评论 26
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值