- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我通常在需要时将 diffutil
与 Recyclerview
一起使用。现在我有这样的情况,我从后端得到的项目是这样的:
data class CarouselItem(var url: String, var pictureUrl: String, var visible: String)
2 个项目的所有 3 个字段可以相同。我在想在这里创建 diffUtil 的最佳方法是什么。恕我直言,我应该制作一些包装器来添加一个字段,通过它我可以区分 2 个项目(例如 ID 字段)。同样在这种情况下,在我的 diffUtil
中,覆盖函数 areContentsTheSame
必须像这样比较项目的所有 3 个字段:
override fun areContentsTheSame(oldItemPosition: Int, newItemPosition: Int): Boolean {
return oldList[oldItemPosition].visible == newList[newItemPosition].visible && oldList[oldItemPosition].pictureUrl == newList[newItemPosition].pictureUrl &&
oldList[oldItemPosition].url == newList[newItemPosition].url
}
我认为这占用的处理器能力与我不使用 DiffUtil
时几乎相同,因为它会将整个项目与整个项目进行比较(仅比较 1 个字段没有任何好处)。
这是正确的做法吗?在这种情况下是否有点矫枉过正?我想知道当我将来遇到这样的情况(像这样的项目)时,作为 android 开发人员的最佳选择是什么? (我喜欢遵循干净的代码概念,并尽可能以最好的方式做事)
最佳答案
由于您使用的是数据类,areContentsTheSame()
可以简化为简单地进行相等性检查:
override fun areContentsTheSame(oldItemPosition: Int, newItemPosition: Int): Boolean =
oldList[oldItemPosition] == newList[newItemPosition]
这种比较通常是微不足道的。相互检查所有项目的目的是避免 notifyDataSetChanged()
的核选项,这会导致对每一行中所有 View 的布局进行完全重做。与属性比较相比, View 布局刷新非常非常昂贵。 (想一想它是如何计算要在二维中显示的文本的每个字符的像素大小并将其全部加起来等等。而比较两个字符串是一种非常快速的字节比较。)
即使与 View 布局更改相比,相等性检查并非微不足道,请记住 DiffUtil 在后台线程上执行相等性检查,但 View 布局刷新必须在主线程上完成,因此使用 DiffUtil 会阻止 UI 感觉简陋。
现在,如果您的两个列表没有任何共同点,DiffUtil 将报告这一点,并且无论如何都必须对所有 View 进行布局,就像使用 notifyDataSetChanged()
一样。如果列表的全部内容都被每次更改完全替换,DiffUtil 可能不是一个好的选择。如果您正在使用 RecyclerView 进行某种分页,可能就是这种情况。
对于用于 DiffUtil 的项目,您应该使用 val
而不是 var
的原因是为了防止误用。 var
可以正常工作,只要您不将它用于列表中的项目,然后期望能够使用 DiffUtil 比较新旧列表。如果您更改了旧列表中的项目,areContentsTheSame()
将毫无意义。这可能会导致一行的 View 在应该更新的时候没有得到更新。
至于您关于唯一 ID 的问题,您可以根据需要生成它们,但最好的方法是视情况而定。 (你的数据是如何存储的,它是如何改变的,什么时候改变的,什么应该保持不变等等。)你的项目类中应该至少有一个属性可以用于 areItemsTheSame()
'比较。您可以省略它,只用与 areContentsTheSame()
相同的方式比较这两个项目,但是您失去了对内容已更改但仍表示相同事物的项目进行适当动画的好处.
关于特定项目情况下的 Android DiffUtil,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/70595104/
我们正在尝试对包含来自 sqlite 数据库的数据的 ArrayList 进行排序。 我可以通过添加到查询“ORDER BY”+ Col_IDBG+“DESC 来对列表进行排序 除了 Recycler
我通常在需要时将 diffutil 与 Recyclerview 一起使用。现在我有这样的情况,我从后端得到的项目是这样的: data class CarouselItem(var url: Stri
我正在尝试使用DiffUtil来更新RecyclerView。我有一个类 DataProvider ,它是一个单例类,它保存包含对象的列表。 每次我需要更新RecyclerView时,我都会修改Dat
我有一个 RecyclerView,其中的元素通过 WebSocket 更新。此更新可能会影响列表中不同位置的多个元素。我想用 DiffUtil更新 RecycleView 的元素。套接字更新本身不包
DiffUtils.dispatchUpdatesTo(adapter) 没有更新我在 recyclerView 中的数组大小 我的 result 的大小大于当前的 itemList,但只显示当前的大
我为 recyclerView 制作了一个适配器,并使用 DiffUtil 以更漂亮的方式显示列表的更新。我使用 google codelabs 作为引用。但该列表并未从以下代码更新。请帮忙 clas
我正在使用 diff util 来提高我的 recyclerview 的性能,而不是调用 notifyDataSetChanged()。 recyclerview 有一个带有一些芯片的标题,可以通过
我有一个具有 DiffUtil 功能的 RecyclerView 适配器。 我想在 DiffUtil 完成其魔法时使用 DataObserver 通知 fragment 。但是,看起来 DiffUti
我的回收站 View 中有无限滚动,因此,它会在有新数据时更新。我正在使用 DiffUtil 更新 recyclerview 中的数据。 DiffUtil 确实会更新数据,但是只要有更新数据,recy
一段时间以来,我一直被困在一个相当简单的问题上。出于某种原因,我无法在 StackOverflow 上找到这方面的答案,所以我想我应该直接寻求帮助。 问题很简单,我有一个按日期排序的交易对象列表。我正
我正在使用下面的代码来测试 DiffUtil实现。根据文档- DiffUtil is a utility class that can calculate the difference between
更新:其中一个问题解决了:现在updateList解决了,问题是我把mAdapter定义为RecyclerView .Adapter 而不是 MyAdapter。但是现在即使我正在获取数据,列表中也没
目前我尝试在我的项目中使用DiffUtil来通知RecyclerView的项目。 与此同时,我编写了一个可以添加页眉和页脚的自定义适配器。我使用了这个适配器并添加了一个加载更多的页脚,这样我的列表在向
我对 Android 还很陌生,目前正在开发我的第一个基于 RecyclerView 的严肃应用程序。当我插入或更新现有行时,我注意到我的应用程序中有一些有问题的行为:即使我在添加或 notifyIt
我使用 ListAdapter 的绑定(bind)和 DiffUtil.ItemCallback 的定义。删除项目(至少 2 个)时出现 IndexOutOfBoundsException。列表的更新
每次我调用无效数据时,我的 DIFF_UTIL 都不会被使用。日志未显示,整个列表已更新为新数据,导致屏幕移动位置等。不确定这里的问题是什么。 我有PagedListAdapter与 LiveData
我无法将 Kotlin Flows 和异步 DiffUtil 放在一起。 我的 RecyclerView.Adapter 中有这个函数,它在计算线程上计算 DiffUtil 并将更新发送到主线程上的
我是 Android 的新手,我正在尝试使用新架构组件实现条码阅读器场景。 每次读取条码时,如果条码不在列表中,我想更新 ViewModel 中的列表,添加新元素,否则增加数量。 以下解决方案有效,但
我遇到了一个有趣的情况,我的适配器有 2 种 View 类型 - HEADER 和 ITEM。 header View 始终位于位置 0。更新数据时,我使用 DiffUtil 获取数据的差异。为了分派
我的 diffcallback areContentsTheSame(oldItem: ItemModel, newItem: ItemModel) 总是接收相同的内容。我用状态来检查,但每次状态都是
我是一名优秀的程序员,十分优秀!