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

标签 android kotlin notifydatasetchanged android-filterable

我正在改进我的应用程序稳定性和性能,但现在我遇到了来自 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在这种情况下(使用过滤器)。在这种情况下,什么是正确的方法以及如何使用它?

最佳答案

据我所知,将 Filterable 接口(interface)与 RecyclerView.Adapter 一起使用是没有意义的。 Filterable 旨在用于 AdapterView 适配器,因为有一些小部件可以检查适配器是否是可过滤的,并且可以自动提供一些过滤功能。但是,RecyclerView.Adapter 与 AdapterView 的 Adapter 没有任何关系。
如果你愿意,你仍然可以使用 Filter 接口(interface)来组织你的代码,但对我来说,这似乎是不必要的额外样板。我在 StackOverflow 上看到其他旧答案说要在 RecyclerView.Adapter 中实现 Filterable,但我认为他们这样做是出于使用旧 Adapter 类的习惯。
至于在过滤时提高适配器的性能,有几个选项。

  • 使用 SortedList 和 SortedList.Callback 来管理您的列表。回调让您实现一堆函数来通知特定项目或项目范围的更改,而不是一次通知整个列表。我没有用过这个,而且似乎有很大的空间出错,因为要实现的回调函数太多了。这也是大量的样板。 top answer here描述了如何做到这一点,但它已经有几年的历史了,所以我不知道是否有更新的方法。
  • 从 ListAdapter 扩展您的适配器。 ListAdapter 的构造函数采用 DiffUtil.ItemCallback 参数。回调告诉它如何比较两个项目。只要您的模型项具有唯一的 ID 属性,这很容易实现。使用 ListAdapter 时,您无需在类中创建自己的 List 属性,而是让父类(super class)处理它。然后,而不是设置一个新的过滤列表并调用 notifyDataSetChanged() ,您调用adapter.submitList()使用您的过滤列表,它使用 DiffUtil 仅自动更改必要的 View ,并且它也具有漂亮的动画。请注意,您不需要覆盖 getItemCount()要么因为父类(super class)拥有该列表。

  • 由于您正在过滤项目,因此您可能希望保留一个额外的属性来存储原始未过滤列表并在应用新过滤器时使用它。所以我在这个例子中创建了一个额外的列表属性。您需要注意仅使用它传递给 submitList()并始终使用 currentListonBindViewHolder()自从 currentList是适配器实际用于显示的内容。
    我删除了 Filterable 函数并使它的外部类可以简单地设置 filter属性(property)。
    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)
    }
    

    关于android - 如何避免可过滤适配器上的 notifyDataSetChanged?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/69526911/

    相关文章:

    android - 如何在我的 Realm 迁移代码中指定一个盒装字段可以为空?

    android - 从我的适配器中刷新 ExpandableListView

    Android 的notifyDataSetChanged()示例

    java - 什么时候调用 super.onActivityResult?

    android - Phonegap/Cordova::以编程方式拍照

    java - 刷新 Activity 中的方法

    android - fragment 中的 adapter.notifyDataSetChanged()

    java - 将击键发送到服务器的 Android 按钮

    android - kotlin.NotImplementedError: EditText 上的 addTextChangedListenerher 未实现操作

    android - 在主线程外运行时对 Cloud Firestore 进行同步调用