我正在改进我的应用程序稳定性和性能,但现在我遇到了来自 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 类的习惯。
至于在过滤时提高适配器的性能,有几个选项。
notifyDataSetChanged()
,您调用adapter.submitList()
使用您的过滤列表,它使用 DiffUtil 仅自动更改必要的 View ,并且它也具有漂亮的动画。请注意,您不需要覆盖 getItemCount()
要么因为父类(super class)拥有该列表。由于您正在过滤项目,因此您可能希望保留一个额外的属性来存储原始未过滤列表并在应用新过滤器时使用它。所以我在这个例子中创建了一个额外的列表属性。您需要注意仅使用它传递给
submitList()
并始终使用 currentList
在 onBindViewHolder()
自从 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/