使用DiffUtil高效更新RecyclerView(转)

DiffUtil是recyclerview support library v7 24.2.0版本中新增的类,根据Google官方文档的介绍,DiffUtil的作用是比较两个数据列表并能计算出一系列将旧数据表转换成新数据表的操作。这个概念比较抽象,换一种方式理解,DiffUtil是一个工具类,当你的RecyclerView需要更新数据时,将新旧数据集传给它,它就能快速告知adapter有哪些数据需要更新。
那么相比直接调用adapter.notifyDataSetChange()方法,使用DiffUtil有什么优势呢?它能在收到数据集后,提高UI更新的效率,而且你也不需要自己对新老数据集进行比较了。
顾名思义,凡是数据集的比较DiffUtil都能做,所以用处并不止于更新RecyclerView。DiffUtil也提供了回调让你可以进行其他操作。本文只讨论使用DiffUtil更新RecyclerView。
DiffUtil简介
在使用DiffUtil前我们先简单看看DiffUtil的特性。DiffUtil使用Eugene W. Myers的Difference算法来计算出将一个数据集转换为另一个的最小更新量,也就是用最简单的方式将一个数据集转换为另一个。除此之外,DiffUtil还可以识别一项数据在数据集中的移动。Eugene的算法对控件进行了优化,在查找两个数据集间最少加减操作时的空间复杂度为O(N),时间复杂度为O(N+D2)。而如果添加了对数据条目移动的识别,复杂度就会提高到O(N2)。所以如果数据集中数据不存在移位情况,你可以关闭移动识别功能来提高性能。
当然这些算法都是封装好的,使用时并不需要关注。下面是谷歌官网给出的在Nexus 5X M系统上进行运算的时长:

  1. 100项数据,10处改动:平均值0.39ms,中位数:0.35ms。
  2. 100项数据,100处改动: a. 打开了移位识别时:平均值:3.82ms,中位数:3.75ms。 b. 关闭了移位识别时:平均值:2.09ms,中位数:2.06ms。
  3. 1000项数据,50处改动: a. 打开了移位识别时:平均值:4.67ms,中位数:4.59ms。 b. 关闭了移位识别时:平均值:3.59ms,中位数:3.50ms。
  4. 1000项数据,200处改动: a. 打开了移位识别时:平均值:27.07ms,中位数:26.92ms。 b. 关闭了移位识别时:平均值:13.54ms,中位数:13.36ms。
    当数据集较大时,你应该在后台线程计算数据集的更新。这一点在后面的代码中会再次说到。
    使用方式
    使用DiffUtil时涉及以下几个核心类:
  5. DiffUtil.Callback:这是最核心的类,不要被命名困惑,它不像你日常所使用的回调。你可以将它理解成比较新老数据集时的规则
  6. DiffUtil:通过静态方法DiffUtil.calculateDiff(DiffUtil.Callback)来计算数据集的更新。
  7. DiffResult:是DiffUtil的计算结果对象,通过DiffResult.dispatchUpdatesTo(RecyclerView.Adapter)来进行更新。
    所以使用步骤如下:
  8. 自定义类继承DiffUtil.Callback,通过覆盖特定方法给出数据比较逻辑
  9. 调用DiffUtil.calculateDiff(DiffUtil.Callback callback[, boolean detectMove])来计算更新,得到DiffResult对象。第二个参数可省,意为是否探测数据的移动,是否关闭需要根据数据集情况来权衡。当数据集很大时,此操作可能耗时较长,需要异步计算。
  10. 在UI线程中调用DiffResult.dispatchUpdatesTo(RecyclerView.Adapter),而后Adapter的onBindViewHolder(RecyclerView.ViewHolder holder, int position, List payloads)。注意这个方法比必须覆盖的onBindViewHolder(RecyclerView.ViewHolder holder, int position)方法多一个参数payloads,而里面存储了数据的更新。

放码
在这里我们使用的数据模式是Item,有四个属性,其中一个是id。我们在这里的逻辑是,根据id判断两个Item对象是不是一项数据,如果是一项数据,则根据Item.equals()方法判断是否这项数据是否被更新。
下面按照上文给出的三个步骤,给出示例代码。
1. 继承DiffUtil.Callback。


除了最后一个getChangePayload()方法,其他都很好理解。最后一个方法的调用情况是:areItemsTheSame()返回true而areContentsTheSame()返回false,也就是说两个对象代表的数据是一条,但是内容更新了。在getChangePayload()方法中,你要给出具体的变化。这里我使用的Bundle,具体使用什么方式来表示数据的更新并不重要,重要的是在这个方法中你把更新情况存入一个对象后,在后面还能从同一个对象中把更新的情况取出来。
2. 计算数据更新情况
这就很简单了,我在收到新数据后新建了一个Runnable来计算,并把得到的DiffResult对象发送到Handler。
3. 将结果发送给Adapter并更新UI
首先我在handler中将数据发送给adapter:

然后重写RecylerView.Adapter.onBindViewHolder(RecyclerView.ViewHolder holder, int position, List payloads)方法,通过payloads.get(0)获取到在DiffUtil.Callback.getChangePayload()方法中返回的Bundle,并取出数据更新情况以更新UI。
结语
DiffUtil可用于高效进行RecyclerView的数据更新,但DiffUtil本身的作用是计算数据集的最小更新。DiffUtil有强大的算法支撑,可以利用DiffUtil完成许多其他功能。
参考文献:
DiffUtil Reference(https://developer.android.com/reference/android/support/v7/util/DiffUtil.html)
DiffUtil is a Must!(https://medium.com/@nullthemall/diffutil-is-a-must-797502bc1149)

原文链接:
http://mp.weixin.qq.com/s?__biz=MzA5ODc5OTI1NQ==&mid=2653425352&idx=1&sn=4828368f722bfb3ea25e9dbc604de7b4&scene=1&srcid=0823gtc36JRwsprGtfo1Ah1Z#rd

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

推荐阅读更多精彩内容