Flutter 分帧上屏源码浅析

2021-07-05  本文已影响0人  _烩面_

最近技术早读会接触了一下 Flutter 优秀的开源库 Keframe, github 链接 点这里 。下面是 github 中作者对于 Keframe 用途的描述:

这是一个通用的流畅度优化方案,通过分帧渲染优化由构建导致的卡顿,例如页面切换或者复杂列表快速滚动的场景。

上面分帧渲染我故意加了粗体。分帧渲染,不明觉厉!趁着周末学习了一下。

关于分帧上屏的原理,这里借用作者 Nayuta 的描述,如下:

假设,我们屏幕能显示4个item,每个item构建耗时是10ms。在现有的 ListView布局过程中,会在第一帧的时候,同时构建这四个item,总共40ms。

采用分帧之后,在页面的第一帧我们先通过构建简单的占位item,占位的item可以是个简单的Container。由于其构建基本不耗时,在第一帧的时候构建四个Container不会导致卡顿。之后将实际的四个item,分别延迟到后面四帧进行渲染。这样对于每个16.7ms而言,都没有发生超时渲染,整个流程不会发生卡顿。

建议上面的话读三遍,就着下面的原理图细品两分钟。

下面就着 Keframe 源码分析一下 分帧是如何实现的

以下是使用 Android Studio 打开之后,Keframe 的工程目录结构。下方黄色框内即为 Keframe 库的核心代码,上方蓝色框内为使用了分帧技术的示例 ListView。

这里我们只看分帧实现的一个核心流程,其它细节暂且忽略掉。所以只分析其中一个示例即可。双击打开 list_opt_example1.dart 文件,下面是构建 ListView 的核心代码

可以看到 ListView 的 itemBuilder 是使用 FrameSeparateWidget 构建的,其构造参数分别是为 index,placeholder,child。传入的参数分别为 i,Container 对象,CellWidget 对象。注意,这里的 placeHolder 即为上面作者叙述帧原理中的占位 item,child 传入的参数则是实际的 item。最终,我们会用实际的 item 去替换占位的 item。Let's move on !

点进 FrameSeparateWidget ,看到的是一个 const 类型的构造函数

const FrameSeparateWidget({
    Key? key,
    this.index,
    required this.child,
    this.placeHolder,
  }) : super(key: key);

同时可以发现 FrameSeparateWidget 继承自 StatefulWidget,重点看一下其状态实现类 _FrameSeparateWidgetState_FrameSeparateWidgetState 重写了父类的 initState(),didUpdateWidget(FrameSeparateWidget oldWidget),build(BuildContext context) 三个方法,同时还实现了一个很重要的方法transformWidget()

先来看 initState() 方法

方法有点长,核心实现已经用黄色框圈出。第一个黄色框内是一个赋值表达式,result 用于接收外界专入的 placeholder,用于占位 item 的渲染(具体可以自己查看 _FrameSeparateWidgetStatebuildContext 的实现),第二个黄色框内从函数名上看是替换 widget 。对,这个就是实际的 item 替换占位 item 的入口!

void transformWidget() {
    SchedulerBinding.instance!.addPostFrameCallback((t) {
      FrameSeparateTaskQueue.instance!.scheduleTask(() {
        if (mounted)
          setState(() {
            result = widget.child;
          });
      }, Priority.animation, id: widget.index);
    });
  }

上面函数里面的实现语句没有多余的,都是重点,需要一句一句的读!首先最外面调用的是系统的任务绑定类 SchedulerBindingaddPostFrameCallback方法。根据系统的方法说明我们知道,该方法会在一个帧渲染开始前向帧前调度队列(这是我自己翻译的) _postFrameCallbacks 里面添加一个任务。addPostFrameCallback 方法具体实现如下

void addPostFrameCallback(FrameCallback callback) {
    _postFrameCallbacks.add(callback);
}

很简单,就是简单把任务 callback 加入到 _postFrameCallbacks 队列中去。在 binding.dart 文件中搜索 _postFrameCallbacks,可以发现在 handleDrawFrame方法里面遍历执行了 _postFrameCallbacks 队列中的任务,从而开始执行 addPostFrameCallback 我们传入的 callback 里面的执行语句。

如上 _invoikeFrameCallback(callback, _currentFrameTimeStamp!),感兴趣的可以自己点进行去看看具体实现。Move on !

接下来看一下我们向 addPostFrameCallback 函数传入的 callback 具体的执行语句。 FrameSeparateTaskQueue 是自己维护的一个任务管理队列类,这里直接调用了 scheduleTask 方法。也传入了一个三个参数:**匿名函数,priority,index **。其中匿名函数中的实现很重要:调用了 setState 方法,并将 widget.child 赋值给了 result。我们知道调用 setState 方法意味着 widget 的绘制,这里应该就是作者叙述的使用实际 widget 替换占位 widget 的具体代码实现。


现在就差扣动执行这个替换的 trigger 了!
接下来就开始 FrameSeparateTaskQueue 类的表演了。
Future<T> scheduleTask<T>(TaskCallback<T> task, Priority priority,
      {String? debugLabel, Flow? flow, int? id}) {
    final TaskEntry<T> entry =
        TaskEntry<T>(task, priority.value, debugLabel, flow, id: id);
    _addTask(entry);
    _ensureEventLoopCallback();
    return entry.completer.future;
  }

首先是实例化了一个 TaskEntry 对象 entry,然后将任务通过 _addTask 方法将 entry 添加到队列里面去。之后通过 _ensureEventLoopCallback 方法开启任务执行时机的监听,以便去执行添加队列中的 entry 任务。那显然得看一下 _ensureEventLoopCallback 方法。

void _ensureEventLoopCallback() async {
    assert(_taskQueue.isNotEmpty);
    if (_hasRequestedAnEventLoopCallback) return;
    _hasRequestedAnEventLoopCallback = true;
    await SchedulerBinding.instance!.endOfFrame;
    _runTasks();
  }

注意倒数第二句的 await SchedulerBinding.instance!.endOfFrame 方法。可以看到 等待每一帧渲染结束 的时候,才会去执行 _runTasks() 方法。见名知意,_runTasks 就是执行任务的意思。继续往下看

void _runTasks() async {
    _hasRequestedAnEventLoopCallback = false;
    bool result = await handleEventLoopCallback();
    if (result)
      _ensureEventLoopCallback();
    else {
      if (_taskQueue.isNotEmpty) {
        _ensureEventLoopCallback();
      }
    }
  }

这里核心的语句是 handleEventLoopCallback() 方法。在该方法中,从队列中取出第一个元素判断其执行优先级:如果优先级足够高,就会执行任务 entry 的 run 方法,接下来返回队列是否为空的布尔值。如果优先级不够,则直接返回 false。返回值用途可以继续查看 _runTasks() 方法关于返回值 result 的使用,用于继续循环监听任务执行的时机。下面是 handleEventLoopCallback() 方法的实现。

方法好长,好烦!不过去除一大堆无关逻辑,核心执行语句也就上面黄色框内的三条语句。到这里,还是没有看到 trigger 的扣动。不过看到了 entry.run(),一丝结束的希望,继续点进去看吧。

终于,来到了最后扣动 trigger 的时刻!

completer.complete(task()) 就是这个 trigger ! task() 的执行意味着实际 widget 替换占位 widget 的真正开始,接下来的实际 widget 的渲染就由系统来完成了!如下,回到了开始的 setState() 方法的执行。

到此,一次完整的分帧上屏任务就完成了!

以上只是就着源码简单分析了一下分帧上屏的实现的核心流程,中间省略了很多的细节,尤其是 Flutter 渲染背后的一套复杂的原理。如果想了解更多,可以跟着作者的思路,一点点的探索分帧上屏诞生的一个过程,感受一下整个过程的艰辛!

强烈推荐阅读: https://juejin.cn/post/6940134891606507534
再放一下 Keframe 地址:Keframe

最后
熟真的能生巧!

再最后
坚持很难,但坚持无价!

上一篇 下一篇

猜你喜欢

热点阅读