- 原文地址:Using leanback’s DiffCallback: The difference between the DiffUtil callbacks
- 原文作者:Benjamin Baxter
- 译文出自:掘金翻译计划
- 本文永久链接:github.com/xitu/gold-m…
- 译者:LeeSniper
- 校对者:tanglie1993, hanliuxin5
24.2 版本的 support library 里引入了一个叫做 DiffUtil 的类,它让刷新 RecyclerView.Adapter 变得更简单。在 27.0 版本的 leanback support library 里面又增加了一个支持 ArrayObjectAdapter 的抽象 DiffUtil
。
ArrayObjectAdapter 有一个新的方法叫做 setItems(final List itemList, final DiffCallback callback),它接收一个新的类叫做 DiffCallback。DiffCallback
看上去很像 DiffUtil.Callback,只是少了几个方法。
public abstract class DiffCallback<Value> {
public abstract boolean areItemsTheSame(@NonNull Value oldItem,
@NonNull Value newItem);
public abstract boolean areContentsTheSame(@NonNull Value oldItem,
@NonNull Value newItem);
@SuppressWarnings("WeakerAccess")
public Object getChangePayload(@NonNull Value oldItem, @NonNull Value newItem) {
return null;
}
}
获取 list 大小的方法不见了!这个 adapter 里的 setItems()
方法知道旧的数据和新的数据,当 adapter 创建 DiffUtil.Callback
的时候,它重写了 getOldListSize() 和 getNewListSize() 方法,让你能够专心比较 list 中数据的异同。
val diffCallback = object : DiffCallback<DummyItem>() {
override fun areItemsTheSame(oldItem: DummyItem,
newItem: DummyItem): Boolean =
oldItem.id == newItem.id
override fun areContentsTheSame(oldItem: DummyItem,
newItem: DummyItem): Boolean =
oldItem == newItem
}
itemsAdapter.setItems(randomItems(), diffCallback)
Adapter 刷新 item 并且播放动画。
ArrayObjectAdapter 会播放合适的动画。
你不一定要调用带有 DiffCallback
的 setItems()
方法。如果你不支持 DiffCallback
,adapter 会清空当前的 item 并且添加所有新的 item,这可能导致你的内容在屏幕上闪一下。
这一行里的内容会在删除和添加 item 的时候闪动。
通过查看 setItems()
的源码,我们可以发现 ArrayObjectAdapter
是如何抽象 DiffUtil
里的样板方法,给开发者提供一个更整洁的 API。
ArrayObjectAdapter 里面 setItems()
方法的部分源码。
如果你想尝试使用 DiffCallback
,可以从参考这篇 gist 开始。
如果你在开发 Android TV 平台上的应用,我很想了解开发过程中你最喜欢的是什么,还有你的痛点是什么。如果你想继续这个话题,请在 Twitter 上给我评论或者留言。