本文主要介绍Android系统中提供的工具类DiffUtil,DiffUtil的主是用与RecyclerView的局部更新,从而提高页面刷新效率。
本文基于最新的v7-27.1.1版本的RecyclerView做介绍,因为这个兼容包里面新增了一个ListAdapter,一并做介绍。
DiffUtil
DiffUtil是在安卓7.0上引入的工具类,相对于“传统”的nofityDataChange()方式:
- DiffUtil可以让我们做到数据局部刷新。虽然我们也可以手动去记录数据改变的位置,自己通过notifyItemRangeInserted等方法去更新数据,但是麻烦~
- DiffUtil可以在局部数据刷新的时候显示动画。
先上demo,看一下DiffUtil要怎么用
简易用法
public static final class MyDiffUtilCallback extends DiffUtil.Callback {
private List mOldList;
private List mNewList;
public MyDiffUtilCallback setDates(List oldList, List newList) {
mOldList = oldList;
mNewList = newList;
return this;
}
@Override
public int getOldListSize() {
return mOldList.size();
}
@Override
public int getNewListSize() {
return mNewList.size();
}
@Override
public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) {
return mOldList.get(oldItemPosition).id == mNewList.get(newItemPosition).id;
// return true;
}
@Override
public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) {
return mOldList.get(oldItemPosition).content.equals(mNewList.get(newItemPosition).content);
}
}
...
public void onNewData(List newData) {
//数据源太大时计算费时,放入子线程
//最新版本兼容包已出ListAdapter,自带子线程执行
DiffUtil.DiffResult diffResult = DiffUtil.calculateDiff(mMyDiffUtilCallback.setDates(mBeans, newData));
diffResult.dispatchUpdatesTo(MyAdapter.this);
mBeans.clear();
mBeans.addAll(newData);
}
上面的代码很简单:
- 就是实现DiffUtil.Callback,指定数据差异的规则
- 在有新数据时,通过DiffUtil.calculateDiff(Callback cb)计算新老数据的差异
- 将计算完的结果更新到adapter中
我们可以看到DiffUtil.Callback主要有4个抽象方法需要实现,看一下这4个方法的具体作用。
public abstract int getOldListSize();
public abstract int getNewListSize();
上面2个方法主要获取新老数据的长度。
//返回值表示新数据传入时这两个位置的数据是否时同一个条目
public abstract boolean areItemsTheSame(int oldItemPosition, int newItemPosition);
//返回值表示新老位置的数据内容是否相同,这个方法在areItemsTheSame()返回true时生效
public abstract boolean areContentsTheSame(int oldItemPosition, int newItemPosition);
上面这两个方法就是区分两个数据bean是否相同。
敲黑板,这里有个注意点:
当areItemsTheSame返回为false时,不管areContentsTheSame是否为true,adapter中的条目都会更新
高级用法
DiffUtil.Callback中还有另外一个可以复写的方法
@Nullable
public Object getChangePayload(int oldItemPosition, int newItemPosition) {
return null;
}
当areItemsTheSame()返回true,同时areContentsTheSame()返回false时,通过这个方法可以用来返回具体的数据差异。
我们来看一下用法
@Override
public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) {
return mOldList.get(oldItemPosition).id == mNewList.get(newItemPosition).id;
return true;
}
@Override
public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) {
return false;
}
@Nullable
@Override
public Object getChangePayload(int oldItemPosition, int newItemPosition) {
Bean oldBean = mOldList.get(oldItemPosition);
Bean newBean = mNewList.get(newItemPosition);
Bundle bundle = new Bundle();
if (!oldBean.content.equals(newBean.content)) {
bundle.putString("content", newBean.content);
}
if (bundle.size() > 0) {
return bundle;
}
return null;
}
然后在Recycler的Adapter中我们通过onBindViewHolder()的另一个重载方法获取到之前设置的数据进行局部刷新。
@Override
public void onBindViewHolder(MyAdapter.ViewHolder holder, int position, @NonNull List payloads) {
if (payloads.isEmpty()) {
super.onBindViewHolder(holder, position, payloads);
//内部其实就是调用的全部刷新的方法
//onBindViewHolder(holder, position);
} else {
Bundle bundle = (Bundle) payloads.get(0);
for (String key : bundle.keySet()) {
if (key.equals("content")) {
holder.contentView.setText(bundle.getString("content"));
}
}
}
}
好了,这里就是DiffUtil的用法了,实现其中的Callback用户比较数据的差异,当来新数据时,调用其中的calculateDiff()静态方法计算数据差异即可。如果你想实现RecyclerView的局部刷新,那可以实现getChangePayload。
ListAdapter
当数据量不大时,我们可以在UI线程中直接更新数据,但是当数据量大时这就比较尴尬了,我们需要自己放在子线程操作,然后再回UI线程更新页面。
在7.0上引入DiffUtil之后,现在又在最新的v7的27.1.1兼容包中加入的官方的支持,就是ListAdapter。
public abstract class ListAdapter
extends RecyclerView.Adapter {
private final AsyncListDiffer mHelper;
@SuppressWarnings("unused")
protected ListAdapter(@NonNull DiffUtil.ItemCallback diffCallback) {
mHelper = new AsyncListDiffer<>(new AdapterListUpdateCallback(this),
new AsyncDifferConfig.Builder<>(diffCallback).build());
}
@SuppressWarnings("unused")
protected ListAdapter(@NonNull AsyncDifferConfig config) {
mHelper = new AsyncListDiffer<>(new AdapterListUpdateCallback(this), config);
}
/**
* Submits a new list to be diffed, and displayed.
*
* If a list is already being displayed, a diff will be computed on a background thread, which
* will dispatch Adapter.notifyItem events on the main thread.
*
* @param list The new list to be displayed.
*/
@SuppressWarnings("WeakerAccess")
public void submitList(List list) {
mHelper.submitList(list);
}
@SuppressWarnings("unused")
protected T getItem(int position) {
return mHelper.getCurrentList().get(position);
}
@Override
public int getItemCount() {
return mHelper.getCurrentList().size();
}
}
ListAdapter用法就跟普通的RecyclerView并无太大差别。
- 一共2个构造方法,区别在于第二个构造方法可以自己指定执行的线程。
- submitList(),用于提交新数据,更新UI。
- 数据比较操作的线程操作在AsyncListDiffer中实现,代码也比较简单,这里就不再贴代码分析了。需要注意点一点时,通过AsyncListDiffer返回的List是一个UnmodifiableList,意味着不能改变长度。
总结时间
- DiffUtil提供了梅耶斯算法实现的新老数据比较的方法,数据比较的规则在Callback
中让用户自行实现,同时实现getChangePayload配合Adaper中的onBindViewHolder(MyAdapter.ViewHolder holder, int position, @NonNull List payloads)可以实现数据的局部更新。 - ListAdapter是对RecyclerView传统Adapter的一个拓展,在子线程中利用DiffUtil比较数据,并在UI线程更新。