Android-RecyclerView系列 RecyclerView滑动后数据显示错乱

一、问题分析

由于RecyclerView执行的onBindViewHolder()方法,在getItemViewType()返回类型不同时会调用,所以如果想要每次都调用onBindViewHolder()刷新item数据,就要重写getItemViewType(),让其返回position,否则很容易产生数据错乱的现象。

二、场景再现

(1)准备一个模拟的数据对象

public class DataModel{

    public DataModel(String description) {
        this.description = description;
    }
    
    public DataModel(String description, boolean selected) {
        this.description = description;
        this.selected = selected;
    }

    public String description;
    public boolean selected;
}

要求:

将15个TestModel对象展示在RecyclerView中,每个Item包含一个TextView和一个ImageView,其中TextView展示description,当selected = true的时候,ImageView显示为选中状态,否则为未选中状态。

(2)准备10个数据,且仅第一个model的selected = true。

List<DataModel> models = new ArrayList<>();
        for (int i = 0; i < 15; i++) {
            TestModel model = new DataModel(String.format("第%s个Item", i));
            models.add(model);
        }
        models.get(0).selected = true;
<ImageView
     android:id="@+id/iv_selected"
     android:layout_width="24dp"
     android:layout_height="24dp"
     android:src="@drawable/unselect"
     android:layout_marginStart="10dp"/>

给ImageView指定了一张默认的图片,当selected = true时再给ImageView指定选中的图片

(3)RecyclerView.Adapter代码

@Override
    public void onBindViewHolder(@NonNull DataAdapter.ViewHolder holder, final int position) {
        DataModel model = models.get(position);
        holder.mTvDes.setText(model.description);
        //指定了选中状态的图片的状态
        if (model.selected) {
           holder.mIvSelected.setImageResource(R.drawable.select_backgound);
        }
    }

效果图-1
在这里插入图片描述
效果图-2 向下滑动
在这里插入图片描述
出现了数据错乱的现象,第12个Item的ImageView中显示了选中图片,打断点发现,该Item也并没有执行if(model.selected)的代码。那么为什么会出现这种现象呢?

三、RecyclerView的复用回收

RecyclerView滚动时,会将已经划出屏幕的ItemView从屏幕上拿下来,放在一个缓存列表中。后续有新的数据需要展示时,从列表中取出一个ItemView用于展示 ,而不是专门重新创建一个ItemView

由于这个ItemView是从缓存中取出的,它依然保留前一次操作后的状态。如果在显示新的数据时,有控件被略过,那么这个控件就会显示上一条数据的状态,从而引起显示上的错乱现象。

四、解决方案

显示错乱的本质是复用引起了控件保留有上次操作的状态 。

1、禁止复用,不过这样就使得性能降低 , 违反了RecyclerView的本质。

// 设置ItemView为不可回收,不能被放入缓存列表,自然无法复用
viewHolder.setIsRecyclable(false);

2、每个ItemView指定不同的类型

在RecyclerView.Adapter内重写下面方法

@Override
    public int getItemViewType(int position) {
        // 给每个ItemView指定不同的类型,这样在RecyclerView看来,这些ItemView全是不同的,不能复用
        return position;
    }

3、保证ItemView中的所有控件都能被刷新。使用条件表达式的方式,对控件的操作考虑完所有的情况

@Override
    public void onBindViewHolder(@NonNull TestAdapter.ViewHolder holder, final int position) {
        DataModel model = models.get(position);
        holder.mTvDes.setText(model.description);
        if (model.selected) {
            holder.mIvSelected.setImageResource(R.drawable.select);
        } else {
            // 保证ImageView控件一定会被操作到
            holder.mIvSelected.setImageResource(R.drawable.unselect);
        }
    }

  • 8
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
Android开发中,我们可以使用RecyclerView来实现网格布局,并且可以通过指定的方式滑动到指定的position。 首先,我们需确保已经在项目的build.gradle文件中添加了RecyclerView的依赖项。 在布局文件中,我们将RecyclerView添加到指定的位置。例如,如果我们希望将RecyclerView添加到activity_main.xml文件的某个LinearLayout中,可以使用以下代码: ```xml <LinearLayout ... android:id="@+id/linear_layout" ...> <androidx.recyclerview.widget.RecyclerView android:id="@+id/recycler_view" ... /> </LinearLayout> ``` 在Activity或Fragment中,我们需要将布局文件中的RecyclerView与代码中的RecyclerView关联起来,并为RecyclerView设置LayoutManager和Adapter。LayoutManager决定了RecyclerView的布局方式,可以选择GridLayoutManager来实现网格布局。Adapter负责为RecyclerView提供数据,并控制每个item显示。 ```java LinearLayout linearLayout = findViewById(R.id.linear_layout); RecyclerView recyclerView = findViewById(R.id.recycler_view); GridLayoutManager layoutManager = new GridLayoutManager(this, 2); // 2代表每行显示2个item recyclerView.setLayoutManager(layoutManager); CustomAdapter adapter = new CustomAdapter(dataList); // 自定义Adapter,提供数据 recyclerView.setAdapter(adapter); ``` 接下来,我们可以通过RecyclerView的smoothScrollToPosition()方法来平滑地滑动到指定的position。例如,如果我们想要滑动到第10个item,可以使用以下代码: ```java recyclerView.smoothScrollToPosition(9); // RecyclerView中position从0开始计数 ``` 这个方法会使RecyclerView平稳地滑动到指定位置,并且会自动滑动到该项的前台,以便用户能够看到该项。 总结来说,我们可以通过在布局文件中添加RecyclerView并设置LayoutManager和Adapter来实现网格布局。然后,通过RecyclerView的smoothScrollToPosition()方法,我们可以滑动到指定的position。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

其子昱舟

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值