速来报名!AICon北京站鸿蒙专场~ 了解详情
写点什么

Flutter 高性能、多功能的全场景滚动容器

  • 2021-01-18
  • 本文字数:5386 字

    阅读完需:约 18 分钟

Flutter 高性能、多功能的全场景滚动容器

目前闲鱼的主要业务场景都已经使用 Flutter 来实现,其中流式布局是最常见的页面布局场景(如搜索、商品详情等)。随着业务的快速迭代和业务复杂度的不断提升,对流式场景的能力和性能要求也越来越高;


  • 在能力方面,最常见的如卡片曝光、滚动锚点、瀑布流布局等能力,随着业务和需求的不断变化,Flutter 原生和一些开源解决方案,渐渐无法满足我们需求。

  • 性能方面,流式场景下的列表滚动流畅度问题随着业务复杂度的增加而逐渐恶化,亟需解决以提升用户的使用体验。


针对以上在业务中面临的问题,我们设计了一套流式场景下通用的页面布局解决方案,我们将其命名为 PowerScrollView。

整体架构设计

在架构设计之前,我们充分调研了原生 Native 的滚动容器:UICollectionView(iOS) 和 RecyclerView(Android)。其中 UICollectionView 的 Section(段落)理念令我们印象深刻,RecyclerView 的架构设计也启发了我们。由于 Flutter 的独特性,我们不能将其照搬过来,所以我们的目标是结合 Native 成熟的滚动容器,加以 Flutter 的特点,设计出更加优秀的滚动容器。


Flutter 原生有常用的 ListView、GridView,他们布局较为单一,功能较为简单。官方也提供了 CustomScrollView 的进阶 Widget,CustomScrollView 由多个 Sliver 进行拼接,以适应更复杂的使用场景,我们将基于 CustomScrollView 进行设计。


从使用角度出发,整个列表由若干个 Section 组成,又将 Section 分为 header、content、footer 三部分,header 为段落的头部,一般可作为 Section 的头部装饰,支持是否吸顶;footer 为段落的尾部,作为 Section 的尾部装饰。列表拥有下拉刷新与加载更多能力;content 为 Section 的正文,支持常见的布局方式:列表、网格、瀑布流以及自定义。Section 的 content 由任意个 cell 组成,cell 即为列表最小粒度的 item。


从 Flutter 原生容器出发,CustomScrollView 支持任意多个 Sliver 的组合,Sliver 提供了 SliverList、SliverGrid、SliverBox 等,已基本符合了我们要求。我们将 Section 的 header 和 footer 各对应一个 SliverBox,content 对应 SliverList 或 SliverGrid,再单独为瀑布流布局开发一个 SliverWaterfall;再在整个列表的头部和尾部插入用于刷新加载更多的 Sliver。


我们将 PowerScrollView 分成数据源管理器、控制器、事件回调和刷新配置四大部分。如下图所示。


null


数据源管理器:用于数据的管理,里面就涉及 Sections 初始化与通常的增删改查。

控制器:主要用于控制 PowerScrollView 的刷新、加载更多,控制滚动到某个位置等。

事件回调:我们将事件分类,外部使用时可只监听需要的回调。

刷新配置:为了提升刷新的灵活性,我们将刷新单独抽出,既可以使用我们提供的标准刷新组建,也可自定义。

功能完善

我们为 PowerScrollView 完善了业务使用的核心诉求,包括自动曝光、滚动到某个 index 、瀑布流、刷新加载更多等能力。下面将重点介绍前两部分。

自动曝光能力

在 Flutter 中,通常不得不将曝光放在 build 函数中,这使得曝光会错乱,不在屏幕上但是在屏幕缓冲区的部分将会被错误曝光,且有多次曝光问题,代码臃肿混乱,这都使得业务层非常头疼。曝光能力是各种业务都必须的核心诉求,我们在 PowerScrollView 中统一进行了封装,通过事件回调给使用者。


前面我们知道,在 PowerScrollView 中,我们用 cell 封装了最小粒度的 item,因为对 item 的封装,使得我们的掌控力大大增强。正因为此,我们自定义了 cell 的 StatefulElement,在 element 的生命周期中 mount、unmount 记录当前 element,利用 InheritedWidget ,将树上的 element 维护在外面的列表中。


在 PowerScrollView 的滚动过程中,我们会遍历检查 element 数组,筛选屏幕中的元素进行曝光回调。其中被筛选掉的即为缓冲区的元素,同时维护个数组避免单元素当次屏幕中多次曝光。


为了减少滚动中的多次遍历检查 element 数组,我们加入了控制滚动采样率的可配参数,通过此参数,我们可以控制滚动一定距离后才进行检查。


在复杂场景中,会存在 cell 高度先为 0,下载模板渲染后再撑开的情况,这种情况下整个 element list 数据会非常大,且数据并不正确,我们需要过滤掉这种。但是当 cell 刷新之后,有了真实的高度,我们需要进行正确的曝光。所以我们在 cell 中监听了 size 的变化,当高度由 0 到非 0 的时候,通知上层进行一次曝光。

滚动到某个 index

Flutter 本身提供了滚动到 position 距离的能力,但一般业务场景下,我们不知道要滚动的距离,最多知道要滚动到第几个,这使得在 Flutter 侧很多交互无法实现。这个问题我们会分几种场景进行分析。


场景一:当要滚动的目标 index 的 cell 在视图树中(当前屏幕及缓冲区),由于我们已经维护了一个屏幕及缓冲区的 element 数组,我们可以遍历找到,然后将其滚动到可见区域即可。


场景二:当要滚动的目标 index 的 cell 不在视图树中时,首先我们根据当前屏幕的 index 与目标 index 进行比较,判断是需要往上滚动还是往下滚动。然后,以较快的速度进行特定距离的滚动,滚动之后再递归,直到找到目标 index。由于滚动距离与时间的不确定性,极端情况下会没有动画效果,普通的动画效果可能也会有些生硬。

性能优化

为什么要做局部刷新

在实际的流式业务场景中,经常会因为数据源的更新而刷新整个列表容器:例如加载了下一页的数据、删除或者插入某一个 cell,甚至某个 cell 的一个按钮状态的变化;


刷新范围过大往往是造成列表容器卡顿、流畅度降低的主要原因,严重影响了用户的操作体验。所以我们需要尽量减少 Widget tree 打脏刷新的范围,减少 Element rebuild 的调用,实现局部刷新的能力。

Viewport 刷新的过程

为什么说整个列表容器打脏刷新会带来严重的耗时呢?我们来简单看一下 Viewport 的刷新过程。


列表容器被打脏之后,会做两个关键的操作:

Viewport 所有 sliver 的 Element 都会 rebuild;

Viewport 也会重新 layout,进而所有的 sliver 也会重新 layout;

我们来先看 Viewport layout 的过程:这个方法的核心,首先找到当前的 center sliver(默认是第一个 child)的位置,然后向上、向下遍历 Viewport 每一个 sliver;每个 child sliver 根据当前 Viewport 在 Scrollview 中的 scrollOffset,Viewport 的大小以及 cacheExtent 大小等信息 (SliverConstraints),计算当前需要展示的 child 的 index 范围,layout 每一个在可显示范围的 child;

以下图例,SliverList 可视范围内需要 layout 的 child index 为 2\~3;SliverGrid 需要 layout 的 child index 为 0\~3;

null

再来看 Viewport 所有 sliver 的 Element rebuild 的过程,这个过程才是列表容器刷新耗时的关键;

我们先来看一下常见的几种布局 SliverList、SliverGrid 以及我们自定义的瀑布流布局 SliverWaterfall 的实现,它们都继承自 SliverMultiBoxAdaptorWidget,一个管理多 child(Box 模型)的 sliver 的基类;它对应的 Element 是 SliverMultiBoxAdaptorElement,主要负责 child 的创建、更新、移除等生命周期相关的工作,这正是局部刷新需要精细处理的地方。


SliverMultiBoxAdaptorElement 内部维护两个 Map,缓存 child element 以及 child widget,在 ViewPort 需要的时候(上面提到的 layout 过程)lazily build 自己的 child;


null


rebuild 过程之所以耗时是因为要清空所有 child widget 缓存,重新 build child widget,update child Element;如果遇到数据的变化,例如 insert、delete,很有可能导致 element 无法复用,这样 rebuild 的成本会更高。

局部刷新的实现原理

摸清了基本原理之后,我们就在思考,当列表容器内容发生变化的时候(比如 insert、delete、LoadMore),是否可以做出一些优化,只让发生变化的部分去 build、layout 呢?


首先我们认为 sliver 的 Element 全部 rebuild 的做法过于简单粗暴,我们可以通过更精准的控制 sliver element 中,childWidgets 与 childElements,来实现局部刷新的目的;


下面我们来看看针对与具体的场景,如何实现精准的 childWidgets 与 childElements 控制,实现局部刷新的能力的。

可变的 child count

在常见的需要局部刷新的场景,容器元素的数量往往会发生变化。在常见的 CustomScrollview 使用中,childCount 都是创建时指定的,当 childCount 方式变化,就需要重新 build 列表容器;

第一步就是避免因为 sliver 内部元素数量变化,必须重新 build 整个容器的问题;

虽然也可以使用 childCount 为空,根据 builder 返回 null 来决定是否为最后一个 child 的方式实现可变 childCount 的目的,但这种方式并不太符合常用的习惯,对使用方也会增加额外成本,所以并未采用这种方式。


做法比较简单,通过继承自 SliverChildBuilderDelegate,修改 childCount 获取方法。


null

局部刷新之 LoadMore

LoadMore 的实现相对会比较简单,需要做的主要有两点:


  1. 清理 widgets 缓存,防止不算加载的过程中内存占用过大;保存与 _childElements 中 index 相同的 widget;这里有一个需要特别注意的点:要过滤为 null 的 widget,否则这个位置的 widget 无法正常展示;(_childWidgets 最后一个 index 会是一个为 null 的值,具体为什么插入一个为 null 的 widget 大家可以阅读源码寻找答案)


null
  1. 最后打脏 sliver,重新 layout children:


null


null


使用 Dart DevTools 的 TimeLine 数据对比两种 LoadMore 方式的耗时情况如下图:

SetState 的 timeline:


null

LoadMore 的 timeline:


null

局部刷新之 Delete

首先整理 childWidgets 的内容,根据 delete 的 index,重新调整 childWidgets 中 widget 与 index 的对应关系;

null

接下来是 _childElements 的处理,如果需要删除的 index 还未创建,只需要把当前 sliver 的 RenderObject 的 layout 信息标脏,重新 layout 自己即可。注意这个过程是不会重新 layout 当前 viewport 已经展示的 child 的


null

否则要找到要删除的 child element,deactivate 对应的 element,其对应的 RenderObject 从 Render tree 上移除:

null

这个过程同时会维护好 child 的 RenderObject 中 ParentData 的 previousSibling 和 nextSibling 的关系;

接下来调整 _childElements 中 Element 与 index 的对应关系;

最后更新每一个 child 的 slot:


null

最后将 sliver 的 RenderObject 标脏,下一帧重新 layout 刷新。

null

局部刷新之 Insert


Insert 的实现过程与上面的类似,可以根据上面的过程自行实现,这里就不做赘述;

Element 复用能力

不管是 iOS 的 UITableView、UICollectionView 还是 Android 的 RecyclerView,都支持 cell 的复用能力;在 Flutter 的列表容器中,在不修改 framework 层的情况下,是否能够实现 element 的复用呢?

首先我们来分析 element 被回收的过程,SliverMultiBoxAdaptorElement 通过 _childElements 来缓存 elements,当滚动超出 viewport 的显示以及预加载范围或者数据源发生变化,会通过调用 collectGarbage 方法回收不需要的 elements;


null

我们可以通过重写 collectGarbage 的方式,在不使用 keepAlive 的情况下,截获本该 deactive 的 child element,放入缓冲池中;在需要创建 element 的时候,优先从缓冲池获取;

虽然原理比较简单,也会遇到一些需要注意的点:需要缓存的 element 需要通过 remove 方法,将它从 childList 中移除,而不是真正的销毁 element, 如果将它被置为 defunct 状态,这样就无法复用了。

因为业务中卡片布局基本相同,这里面复用的逻辑做的相对简单,事实上针对卡片类型复用才能发挥出最好的效果。

分帧渲染

在实际的滑动过程中,如果一帧的时间内需要 build 过多的 cell ,很容易引起掉帧的情况,用户会感觉到卡顿。为了减少这种情况,我们在 cell 层面引入了 placeholder 的机制:


null

使用方可以为每个 item 定制较为简单的 Widget,这样在一帧任务较多时,通过一定的策略,先 build placeholder 进行渲染,延迟到之后几帧再进行实际 cell 的 build。由于 viewport 上下都有缓冲区,在延后的帧设置较少时,用户并没有机会看到 placeholder,所以业务上并不会有影响。placeholder 最明显的作用是削峰,较长的一帧耗时会被下几帧瓜分。


下面数据是使用复杂商品 card 在瀑布流中的场景,使用机型为 Pixel XL。从数据上看,分帧使平均耗时有所增加,但是 90、99、最长帧耗时,都有明显的降低,丢帧数也有所减少。

null

值得注意的是,对于 cell 过于复杂的场景,即使一帧 build 一个都会超时,那么以 cell 为最小粒度的分帧就没有优化效果了,类比到在性能非常差的手机上,普通复杂的 cell 的分帧可能会使流畅度降低。这个时候需要降低 cell 复杂度或者缩小分帧的粒度。

实际应用场景

PowerScrollView 已经在闲鱼多个核心页面线上全量使用,如下图:

null


完善的能力、优良的性能、较低的接入成本,都使得使用方受益颇多。

总结和展望

经过对列表容器能力的不断完善、流畅度方面不断优化,目前 PowerScrollView 已经能够更好的支撑闲鱼流式布局下的业务,给用户提供更好的使用体验。


但在一些低端机型上,长列表的表现仍然不能让人满意;瀑布流等一些需要复杂布局计算的场景,如何更好的优化布局计算过程,这些都是需要我们继续探索的方向。


目前复用实现还比较粗糙,未来也会深入到 Flutter 引擎,寻找提升复用能力的方法,让 PowerScrollView 真正成为一个高效流式布局的解决方案。


另外在端到端研发方面,我们在探索将列表容器与动态模板相结合,实现端云一体的页面搭建解决方案。


本文转载自:闲鱼技术(ID:XYtech_Alibaba)

原文链接:Flutter 高性能、多功能的全场景滚动容器

2021-01-18 07:002713

评论

发布
暂无评论
发现更多内容

传统企业的办公模式正在发生着变化,为什么企业需要数字化办公?

WorkPlus

G1GC算法读书笔记(更新中)

老猎人

为Python打包创建一个世外桃源,解决打包太大且启动慢的问题

迷彩

pyinstaller 7月月更 Python打包

接口文档进化图鉴,有些古早接口文档工具,你可能都没用过

Liam

Postman 接口文档 API swagger API文档

【干货】知识共享的障碍及解决方法

Geek_da0866

行业洞察|如何更好地建设数据中台?IT和业务要“齐步走”

WorkPlus

Review 后台管理系统实战:请求参数的 2 种封装风格

掘金安东尼

前端 编程范式 7月月更

知识分享|分享一些提升企业文档管理水平的方法

Baklib

经验分享|企业该怎样利用SaaS进行企业知识管理

Baklib

李宏毅《机器学习》丨5. Tips for neural network design(神经网络设计技巧)

AXYZdong

机器学习 7月月更

API策略因何成为企业数字化转型的制胜法宝?

WorkPlus

算法题每日一练---第4天:图像模糊问题

知心宝贝

算法 前端 后端 7月月更

DistSQL 深度解析:打造动态化的分布式数据库

SphereEx

数据库 开源社区 ShardingSphere SphereEx #开源

java程序员培训班怎么选?

小谷哥

学好Web前端开发能找到好工作吗

小谷哥

阿里云技术专家郝晨栋:云上可观测能力——问题的发现与定位实践

阿里云弹性计算

DevOps 运维 可观测性

8个方法管理 GitHub 用户权限

SEAL安全

git GitHub 安全 软件安全 软件供应链安全

N分钟学会分位值的计算方式

眼镜盒子

指标

如何快速开发一个简单实用的MES系统?

优秀

MES系统

无套路、无陷阱、无广告 | 这个免费的即时通讯软件确定不用吗?

WorkPlus

等额本金递增还款/等额本金递减按揭房贷还款计算器

入门小站

工具

在线XML转CSV工具

入门小站

工具

京东云分布式链路追踪在金融场景的最佳实践

京东科技开发者

数据库 分布式 京东云 云计算,

linux 上查找包含特定文本的所有文件

入门小站

Linux

“万物互联,使能千行百业”,2022 开放原子全球开源峰会 OpenAtom OpenHarmony 分论坛即将开幕

kk-OSC

开源 开放原子全球开源峰会

学习java开发技术有用吗?

小谷哥

鼓励企业知识共享的好处,你知道多少?

Geek_da0866

图的基本定义和概念(二)

乔乔

7月月更

这样优化Spring Boot,启动速度快到飞起!

艾小仙

Java 微服务 springboot Eureka 微服务治理

大数据培训机构如何选择

小谷哥

学习大数据技术之前做好这些准备

小谷哥

Flutter 高性能、多功能的全场景滚动容器_大前端_闲鱼技术_InfoQ精选文章