如何避免可过滤适配器上的 notifyDataSetChanged?

Posted

技术标签:

【中文标题】如何避免可过滤适配器上的 notifyDataSetChanged?【英文标题】:How to avoid notifyDataSetChanged on a Filterable Adapter? 【发布时间】:2021-11-30 05:54:14 【问题描述】:

我正在改进我的应用稳定性和性能,但现在我遇到了来自 android Studio 的警告。请考虑以下适配器类:

private class CoinsAdapter(private val fragment: CoinFragment, private val coins: List<Coin>): RecyclerView.Adapter<CoinsAdapter.ViewHolder>(), Filterable 

    private val filter = ArrayList(coins)

    override fun onCreateViewHolder(parent: ViewGroup, position: Int): ViewHolder 
        val binding = ItemCoinBinding.inflate(LayoutInflater.from(parent.context), parent, false)
        return ViewHolder(binding)
    

    override fun onBindViewHolder(holder: ViewHolder, position: Int) 
        val coin = filter[position]
        
        holder.binding.coinImage.setImageResource(coin.image)
        holder.binding.coinText.text = builder.toString()
    

    override fun getItemCount() = filter.size

    override fun getFilter() = object : Filter() 

        override fun performFiltering(constraint: CharSequence): FilterResults 
            if (constraint.length < 2) return fetchResults(coins)
            val pattern = constraint.toString().lowercase().trim()

            val filter = arrayListOf<Coin>()
            for (coin in coins) if (coin.name.lowercase().contains(pattern)) filter.add(coin)

            return fetchResults(filter)
        

        private fun fetchResults(coins: List<Coin>): FilterResults 
            val results = FilterResults()
            results.values = coins

            return results
        

        override fun publishResults(constraint: CharSequence, results: FilterResults) 
            filter.clear()
            filter.addAll(results.values as List<Coin>)

            notifyDataSetChanged()
        
    

    private inner class ViewHolder(val binding: ItemCoinBinding) : RecyclerView.ViewHolder(binding.root)

适配器和过滤器工作正常,但请注意publishResults 函数。 Android Studio 就notifyDataSetChanged 发出警告。

It will always be more efficient to use more specific change events if you can. Rely on notifyDataSetChanged as a last resort.

但是,我不知道如何在这种情况下使用notifyDataSetChanged(带有过滤器)。在这种情况下,什么是正确的方法以及如何使用它?

【问题讨论】:

【参考方案1】:

据我所知,将 Filterable 接口与 RecyclerView.Adapter 一起使用是没有意义的。 Filterable 旨在用于 AdapterView 适配器,因为有一些小部件可以检查适配器是否是可过滤的,并且可以自动提供一些过滤功能。但是,RecyclerView.Adapter 与 AdapterView 的 Adapter 没有任何关系。

如果愿意,您仍然可以使用过滤器接口来组织代码,但在我看来,这似乎是不必要的额外样板。我在 *** 上看到其他旧答案说要在 RecyclerView.Adapter 中实现 Filterable,但我认为他们这样做是出于使用旧 Adapter 类的习惯。

关于在过滤时提高适配器的性能,有几个选项。

    使用 SortedList 和 SortedList.Callback 来管理您的列表。回调让您实现一堆函数来通知特定项目或项目范围的更改,而不是一次通知整个列表。我没有用过这个,而且似乎有很大的空间出错,因为要实现的回调函数太多了。这也是大量的样板。 top answer here 描述了如何做到这一点,但它已经有几年的历史了,所以我不知道是否有更新的方法。

    从 ListAdapter 扩展您的适配器。 ListAdapter 的构造函数采用 DiffUtil.ItemCallback 参数。回调告诉它如何比较两个项目。只要您的模型项具有唯一的 ID 属性,这很容易实现。使用 ListAdapter 时,您无需在类中创建自己的 List 属性,而是让超类处理它。然后,您无需设置新的过滤列表并调用notifyDataSetChanged(),而是使用您的过滤列表调用adapter.submitList(),它使用DiffUtil 仅自动更改必要的视图,并且它还带有漂亮的动画。请注意,您也不需要覆盖 getItemCount(),因为超类拥有该列表。

由于您正在过滤项目,您可能希望保留一个额外的属性来存储原始未过滤列表并在应用新过滤器时使用它。所以我在这个例子中创建了一个额外的列表属性。您需要注意仅使用它传递给submitList() 并始终在onBindViewHolder() 中使用currentList,因为currentList 是适配器实际用于显示的内容。

并且我删除了 Filterable 函数并使其成为外部类可以简单地设置 filter 属性。

class CoinsAdapter : ListAdapter<Coin, CoinsAdapter.ViewHolder>(CoinItemCallback) 
    
    object CoinItemCallback : DiffUtil.ItemCallback<Coin>() 
        override fun areItemsTheSame(oldItem: Coin, newItem: Coin): Boolean = oldItem.id == newItem.id
        override fun areContentsTheSame(oldItem: Coin, newItem: Coin): Boolean = oldItem == newItem
    
    
    var coins: List<Coin> = emptyList()
        set(value) 
            field = value
            onListOrFilterChange()
        

    var filter: CharSequence = ""
        set(value) 
            field = value
            onListOrFilterChange()
        

    override fun onCreateViewHolder(parent: ViewGroup, position: Int): ViewHolder 
        val binding = ItemCoinBinding.inflate(LayoutInflater.from(parent.context), parent, false)
        return ViewHolder(binding)
    

    override fun onBindViewHolder(holder: ViewHolder, position: Int) 
        val coin = currentList[position]

        holder.binding.coinImage.setImageResource(coin.image)
        holder.binding.coinText.text = builder.toString()
    

    private fun onListOrFilterChange() 
        if (filter.length < 2) 
            submitList(coins)
            return
        
        val pattern = filter.toString().lowercase().trim()
        val filteredList = coins.filter  pattern in it.name.lowercase() 
        submitList(filteredList)
    

    inner class ViewHolder(val binding: ItemCoinBinding) : RecyclerView.ViewHolder(binding.root)

【讨论】:

我刚刚实施了您的解决方案,它正在创造奇迹。非常感谢您的描述性解释和解决方案!【参考方案2】:

notifyDataSetChanged 重绘整个视图,这就是 Android Studio 显示警告的原因。

要解决这个问题,您可以使用DiffUtil

private class CoinsAdapter(private val fragment: CoinFragment, private val coins: List<Coin>): RecyclerView.Adapter<CoinsAdapter.ViewHolder>(FilterDiffCallBack()), Filterable 
 ....
 ....
  //This check runs on background thread
class FilterDiffCallBack: DiffUtil.ItemCallback<Post>() 
    override fun areItemsTheSame(oldItem: Coin, newItem: Coin): Boolean 
      
        return oldItem.someUniqueId == newItem.someUniqueId
    

    override fun areContentsTheSame(oldItem: Coin, newItem: Coin): Boolean 
        
        return oldItem == newItem
    

...
...
override fun publishResults(constraint: CharSequence, results: FilterResults) 

        submitList(results)// call the DiffUtil internally
    

如果列表中的数据大部分随用户交互而变化,那么您可以使用notifyItemChanged(int)notifyItemInserted(int) notifyItemRemoved(int) 等方法,因为这是更新视图的最有效方法。更多信息here

【讨论】:

整个视图在你滚动的时候被重绘了很多次。 notifyDataSetChanged() 很慢,因为它还会导致重新测量整个布局,而不仅仅是重新绘制。它还会导致闪烁,因为它不执行动画来显示内容如何变化。

以上是关于如何避免可过滤适配器上的 notifyDataSetChanged?的主要内容,如果未能解决你的问题,请参考以下文章

我的可过滤适配器正在查找结果但未显示

如何将SQLite数据与可重用的RecyclerView适配器一起使用

如何在Android Studio中创建入门芯片填充?

搜索结果不匹配时,android中没有数据吐司

可过滤在 Recyclerview 中无法按预期工作

列表视图