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

Rav*_*ers 7 android kotlin notifydatasetchanged android-filterable

我正在提高应用程序的稳定性和性能,但现在我遇到了来自 Android Studio 的警告。请考虑以下 Adapter 类:

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)
}
Run Code Online (Sandbox Code Playgroud)

适配器和过滤器工作完美,但请注意其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在这种情况下使用(使用过滤器)一无所知。在这种情况下什么是正确的方法以及如何使用它?

Ten*_*r04 6

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

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

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

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

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

由于您正在过滤项目,因此您可能需要保留一个额外的属性来存储原始的未过滤列表,并在应用新过滤器时使用它。所以我在这个例子中创建了一个额外的列表属性。您需要小心,仅使用它来传递submitList()并始终使用currentListin onBindViewHolder(),因为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)
}
Run Code Online (Sandbox Code Playgroud)