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
在这种情况下使用(使用过滤器)一无所知。在这种情况下什么是正确的方法以及如何使用它?
据我所知,将 Filterable 接口与 RecyclerView.Adapter 一起使用是没有意义的。Filterable 旨在在 AdapterView 适配器中使用,因为有一些小部件可以检查 Adapter 是否是 Filterable 并可以自动提供一些过滤功能。但是,RecyclerView.Adapter 与 AdapterView 的 Adapter 没有任何关系。
如果您愿意,您仍然可以使用 Filter 接口作为组织代码的方式,但对我来说,这似乎是不必要的额外样板。我在 StackOverflow 上看到过其他旧答案,说要在 RecyclerView.Adapter 中实现 Filterable,但我认为他们这样做是出于使用旧 Adapter 类的习惯。
至于在过滤时提高适配器的性能,有几个选项。
使用 SortedList 和 SortedList.Callback 来管理您的列表。回调让您实现一堆函数来通知特定项目或项目范围的更改,而不是立即通知整个列表。我没有使用过这个,而且似乎有很大的空间出错,因为有太多的回调函数需要实现。这也是大量的样板文件。这里的最佳答案描述了如何做到这一点,但它已经有几年了,所以我不知道是否有更新的方法。
从 ListAdapter 扩展您的适配器。ListAdapter 的构造函数采用 DiffUtil.ItemCallback 参数。回调告诉它如何比较两个项目。只要您的模型项具有唯一的 ID 属性,这就很容易实现。使用 ListAdapter 时,您不会在类中创建自己的 List 属性,而是让超类处理它。然后,notifyDataSetChanged()
您不必设置新的过滤列表并调用 ,adapter.submitList()
而是使用过滤列表进行调用,它使用 DiffUtil 自动仅更改必要的视图,并且它也使用漂亮的动画来完成此操作。请注意,您不需要重写getItemCount()
其中任何一个,因为超类拥有该列表。
由于您正在过滤项目,因此您可能需要保留一个额外的属性来存储原始的未过滤列表,并在应用新过滤器时使用它。所以我在这个例子中创建了一个额外的列表属性。您需要小心,仅使用它来传递submitList()
并始终使用currentList
in 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)
归档时间: |
|
查看次数: |
2558 次 |
最近记录: |