RxJava和DiffUtil结合

原文地址:https://android.jlelse.eu/a-nice-combination-of-rxjava-and-diffutil-fe3807186012

如果你使用RecyclerView,并且保持了与API相同的更新,你也许会注意到DIffUtil类添加了好几个版本,这个优秀的工具类通过将已经存在的数据和新生成的数据进行对比,然后调用notifyItemInsertednotifyItemRemoved来轻松实现数据的改变。你需要做的就是实现一个回调,在回调中将现有数据和新获取的数据进行比较。

private static class MyDiffCallback extends DiffUtil.Callback{
  private List<Thing> current;
  private List<Thing> next;
  public MyDiffCallback(List<Thing> current,List<Thing> next){
    this.current  = current;
    this.next = next;
  }

  @Override
  public int getOldListSize(){
    return current.size();
  }

 @Override
  public int getNewListSize(){
    return next.size();
  }

  @Override
  public boolean areItemsTheSame(int oldItemPosition,int newItemPosition){
    Thing currentItem = current.get(oldItemPosition);
    Thing nextItem = next.get(newItemPosition);
    return currentItem.getId() == nextItem.getId();
  }

  @Override
  public boolean areContentsTheSame(int oldItemPosition,int newItemPosition){
    Thing currentItem = current.get(oldItemPosition);
    Thing nextItem = current.get(newItemPosition);
    return currentItem.equals(nextItem);
  }
}
DiffResult diffResult = DiffUtil.calculateDiff(new MyDiffCallback(current,next),true);
diffResult.dispatchUpdatesTo(adapter);

上述代码显示了如何实现回调,如何执行计算以及如何将notify-call分配到RecyclerView的适配器。

然而这里还存在一个挑战那就是如果数据量特别大的话,或者对比比较复杂的情况下,你不能在主线程中调用计算功能,需要把这些移到一个后台进程,然后把结果发送给主线程用来设置新的数据,

现在事情变得很棘手,因为DiggUtil计算需要用到已经存在的数据和新的数据,如果你的适配器有一个setData()方法,那么他将需要一个getData()方法,这意味着将从多个线程访问数据,因此你需要一些用于同步或者线程安全的数据结构,如何避免这种情况呢。

使用Rxjava将会解决一切问题,假设你有一个Flowable<List<Thing>> listOfThings,他将发出RecycleView将显示的获取的新版本的数据,我们可以在IO或者在计算线程调度以免阻塞主线程,然后再主线程进行观察事件将数据传递给adapter。

ThingRepository
  .latestThings(2,TimeUnit.SECONDS)
  .subscribeOn(computation())
  .observeOn(mainThread())
  .subscribe(things -> {
    adapter.setThings(things);
    adapter.notifyDataSetChanged();
  })

以上的代码确保了在计算线程中获取新的数据列表,并且发送给主线程然后调用adapter的notifyDataSetChanged,这样做有效,但是看起来不是很好,因为每个新列表都需要重新绘制。

为了使用DiffUtil,我们需要调用calculateDiff()方法,并将DiffResult与最新的List<Thing>一起传递给我们的订阅者,一个简单的方法是使用Pair类,这是支持库中一个简单并且强大的实现类,我们将更改订阅以接受Pair<List<Thing>,<DiffResult>>事件。

.subscribe(listDiffResultPair -> {
  List<Thing> nextThings = listDiffResultPair.first;
  DiffUtil.DiffResult diffResult = listDiffResultPair.second;
  adapter.setThings(nextThings);
  diffResult.dispatchUpdatesTo(adapter);
});

.scan()

传递给DiffUtil.calculateDiff()的回调需要当前列表和新列表才能生效,我们如何确保我们从数据源所获取的每一个新列表,我们也想获取上一个事件,这也是RxJava更神秘的一个运算符scan(),scan的初始值将是一个由空列表和空值作为DiffResult值组成的对。

现在我们将会调用calculateDiff()并将其列在我们的Pair中,我们使用结果构建一个新的DiffResult。
我们还有一件事要考虑,如果我们这样离开,第一件事将包含一个DiffResult为null的Pair,所以我们必须在我们的订阅中对null进行检查,但是,由于也有可能我们开始的时候是一个空列表,所以我们可以使用skip运算符来简单的跳过他,这将会忽略第一个事件。

List<Thing> emptyList = new ArrayList<>();
adapter.setThings(emptyList);
Pair<List<Things>,DiffUtil.DiffResult> initialPair = Pair.create(emptyList,null); 
ThingRepository
  .latestThings(2,TimeUnit.SECONDS)
  .scan(initialPair,(pair,next) -> {
    MyDiffCallback callback = new MyDiffCallback(pair.first,next);
    DiffUtil.DiffResult result = DiffUtil.calculateDiff(callback);
    return Pair.create(next,result);
  })
  .skip(1);

这可以使用RxJava和一些职能操作符在后台线程上使用DiffUtil,还没有必要考虑适配器保存的当前数据的任何同步或并发,我们的实力应用程序的结果可以在下面看到。

Demo地址:

https://github.com/ErikHellman/RxJavaAndDiffUtil

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,056评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,842评论 2 378
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,938评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,296评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,292评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,413评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,824评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,493评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,686评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,502评论 2 318
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,553评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,281评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,820评论 3 305
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,873评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,109评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,699评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,257评论 2 341

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,335评论 25 707
  • 我从去年开始使用 RxJava ,到现在一年多了。今年加入了 Flipboard 后,看到 Flipboard 的...
    Jason_andy阅读 5,442评论 7 62
  • 前言我从去年开始使用 RxJava ,到现在一年多了。今年加入了 Flipboard 后,看到 Flipboard...
    占导zqq阅读 9,157评论 6 151
  • MySQL数据库是较为常见的数据,在阿里云上有提供DMS能方便快捷的管理数据库,可以阿里云帮你创建,也可以自己...
    Lazy_Caaat阅读 1,359评论 2 3
  • 罗贯中先生写《三国演义》是注入了感情色彩的。亲蜀疏魏中立于吴。他的情感自然会传递给读者。不过成人读书应客观公正,...
    沉之阅读 850评论 1 6