首页 > 解决方案 > AsyncListDiffer 没有更新 recyclerview

问题描述

我有一个 RecyclerView 和一个使用AssyncListDiffer的适配器。我遇到的问题是,当 LiveData 发生更改时,recyclerview没有更新。观察者收到通知,但列表没有更新。

这是我的适配器:

class HourAdapter(private val interaction: HourInteraction? = null) :
RecyclerView.Adapter<HourAdapter.HourViewHolder>() {

    private val differ = AsyncListDiffer(this, DIFF_CALLBACK)

    fun submitList(list: List<Hour>?) {
        differ.submitList(list)
    }

    private fun getHourAt(position: Int): Hour {
        return differ.currentList[position]
    }

    override fun getItemCount(): Int {
        return differ.currentList.size
    }

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): HourViewHolder {...}

    override fun onBindViewHolder(holder: HourViewHolder, position: Int) {...}

    val DIFF_CALLBACK = object : DiffUtil.ItemCallback<Hour>() {

      override fun areItemsTheSame(oldItem: Hour, newItem: Hour): Boolean {
        return (oldItem.name == newItem.name) && (oldItem.isChecked == newItem.isChecked)
        }

      override fun areContentsTheSame(oldItem: Hour, newItem: Hour): Boolean {
        return oldItem == newItem
       }

     }

    class HourViewHolder
        (
        internal val binder: HourItemBinding
    ) : RecyclerView.ViewHolder(binder.root)

}

我使用“ submitList() ”提交新列表。但它不起作用

我在网上寻找解决方案,基本上有两种选择:

  1. submitList函数中,调用 AssyncListDiffer 的“submitList”两次,如下所示:

        differ.submitList(null)
        differ.submitList(list)
    }```
    
  2. 第二个选项是使用ListAdapter并像这样覆盖“submitList”函数:
  override fun submitList(list: List<Hour>?) {
  super.submitList(list?.let { ArrayList(it) })
  }

第一个解决方案有效,但每当我更新它时,recyclerview 就会闪烁。覆盖该方法的第二种解决方案对我不起作用。

几天来我一直在尝试解决这个问题,但我无法让它工作。另外,我不想使用 notifyItemChanged() 或 notifyDataSetChanged()。

还有其他方法吗?

标签: androidkotlinandroid-recyclerviewandroid-diffutilsandroid-asynclistdiffer

解决方案


我遇到了同样的事情并观察了以下内容。

每次AsyncListDiffer收到我的清单;它与以前的对象相同 - 存在于内存中。因此,不同的决定没有任何改变,也没有提交更新的列表。

我的列表中包含一个对象,并且对于每次提交尝试,我都在更改一个字段。对象和列表当然保持不变。

所以,我想知道为什么选项 2 不起作用,结果我需要更有表现力:

submitList(it.map {
    it.copy()
})

否则,Kotlin 不会制作对象的深层副本。


推荐阅读