Flutter性能优化实践之Timeline

前言

Flutter自诞生之时就以轻松构建美观、高性能组件著称,目标是提供逼近“原生性能”的60帧每秒(fps)的性能,或者是在可以达到120Hz的设备上提供120fps的性能。这里的帧率fps是指的画面每秒传输帧数,是衡量性能优化中屏幕是否卡顿的一个重要指标,如何测量一个应用的帧率,就要用到工具Timeline。

注:关于Performance性能指标的描述有多个方面,本文侧重点为Timeline

Flutter 性能分析性能

flutter 支持三种模式编译的app,处于开发的不同阶段,使用不同模式下的app

调试模式

在命令行下输入flutter run,默认会启动debug模式,该模式下app使用JIT的编译方式,运行时去解析执行程序,意味着应用有着较慢的性能体验,比如冷启动或者第一次初始化Flutter Engine会有较长时间的黑屏、使用过程中掉帧和卡顿这都属于正常现象。

该模式下一个突出的特点是可以热重载,所以更像开发一个前端应用

Release模式

命令行输入flutter run --release 会使用 Release 模式来进行编译,该模式下应用具有最大的优化和性能体验,采用AOT的编译技术,由dart本地虚拟机将代码编译成对应平台例如Android、IOS对应的机器码,相当于原生开发,编译耗时,失去了热重载,但具有良好性能。一般用于最后应用市场发包,失去了debug模式下的各种调试应用功能

Profile模式

命令行输入flutter run --profile会使用Profile模式编译,一个专门的调试应用性能模式,该模式是在保留一部分调试能力的基础上,又较大程度还原app真实性能,所以该模式不建议在虚拟机或模拟器上运行,因为无法真实代表真机性能。

输入该命令,运行完以后控制台会打印调试的地址

image

点击即可跳转到调试页

image

这里选择timeline,点击Flutter Developer按钮,即可进入timeline的调试页面,在手机上操作几秒钟,点击右上角Refresh按钮,即可加载出图像,看页面代码应该也是临时借用的systemtrace的,操作都类似

image

这个其实是以前旧版的调试方式,现在虽然也能使用,但是实际测试中发现不太准确,使用也不方便,现在基本不使用这种方式了,新版本的Performance页面很美观使用也方便,可以在Android studio中使用Flutter Performance插件中页面粗略判断timeline是否卡顿,也可以打开Flutter Performance右下角Open DevTools按钮在网页上具体分析。

Flutter Inspector

这里顺便说下Android Studio中其他两个Flutter 插件,一个是Flutter Outline,即显示页面布局的大纲,可以快速查看页面布局的树形结构,菜单栏提供了包裹、删除、上下移动组件的快捷功能,很简单这里不详细介绍。另一个插件是Flutter Inspector,安装Flutter插件后,AS右侧边栏会出现这个标签,主要作用是开发过程中布局调试用的,类似Android开发里的Xml布局查看工具,只不过需要debug模式运行时才可以查看布局,这点相对于原生开发xml快速定位布局文件的体验还是差一些,相信后期还会有好的优化。

点击标签后页面如下

image

页面主要分上边功能按钮、左边View树、右边布局预览。

每个按钮的作用:

Select Widget Mode

image

选择组件模式,选中后点击下方View Tree中某个Widget自动定位到代码位置,可在开发中快速定位代码

Refresh Tree

image

刷新View Tree,在App上跳转其他页面后,View Tree不自动更新,所以有了此按钮

Slow Animations

image

放慢动画

Debug Paint

image

显示布局测量,可以快速确定组件边界,效果如下


image

Show Paint Baselines

image

显示Text组件的Baseline,方便文字对齐

Show Repaint Rainbow

image

显示重绘时颜色变化

Invert Oversized Images

image

轻松查看分辨率比显示分辨率高的图片

image

Flutter Performance

点击右边栏Flutter Performance,出现如下页面:

image

左上角第一个按钮是在手机上显示performance Overlay,效果如下,其他按钮和上边Inspector中一样

image

上边可以粗略判断App是否掉帧,白色正常,红色就卡顿了,中间内存占用,下边是每个组件的重绘状态,点击右下角的Open DevTools可以使用更多功能

Track widget rebuilds复选框勾上可以方便的查看页面中组件的重绘状态,对于不应该重绘的组件应该调整代码层级结构或者抽离组件的方式避免重绘造成性能的损失,这里分享个人在开发中总结的几点经验:

  • 尽量少使用StatefulWidget编写大的页面,尽量避免在StatefulWidget中使用setState
  • 不需要重绘组件添加const关键字
  • Provider刷新机制时使用Consumer下沉刷新范围
  • 小部件需要刷新抽取成StatefulWidget,缩小刷新范围

Timeline

时间线事件图表显示了应用程序中的所有事件跟踪。 Flutter框架在构建框架,绘制场景以及跟踪其他活动(例如HTTP流量)时会发出时间轴事件。这些事件显示在时间轴上。您还可以通过dart发送自己的时间线事件:developer TimelineTimelineTask API

Timeline 事件轨迹的格式和查看器并被许多其他项目使用,此类项目包括 Chromium & Android (via systrace).

轨迹记录的形式是JSON文件格式存储的,点击右上角的Export按钮可以导出文件。

打开DevTools以后,在App上操作一段,点击左上角Refresh按钮即可加载出如下图所示时间线。


image

图中蓝色条是正常帧,红色条是卡顿帧,鼠标移动到红条上可以查看当前卡顿帧的耗时,右上角有不同颜色条的对应关系,分别有UI、Raster、Jank

UI

UI线程在Dart VM中执行Dart代码。这包括您的应用程序以及Flutter框架中的代码。当您的应用创建并显示场景时,UI线程将创建一个层树(包含与设备无关的绘画命令的轻量级对象),并将该层树发送到要在设备上呈现的栅格线程。不要阻塞该线程。

Raster

光栅线程(以前称为GPU线程)执行Flutter Engine中的图形代码。该线程获取层树并通过与GPU(图形处理单元)对话来显示它。您无法直接访问栅格线程或其数据,但是如果该线程速度很慢,则是由于您在Dart代码中所做的操作所致。图形库Skia在此线程上运行。

有时,场景会产生易于构造的图层树,但是在栅格线程上渲染的树代价很高。在这种情况下,您需要弄清楚代码正在做什么,这会导致渲染代码变慢。对于GPU而言,特定种类的工作负载更加困难。它们可能涉及对saveLayer()的不必要调用,与多个对象相交的不透明性以及在特定情况下的剪辑或阴影

Jank

帧渲染图显示带有红色叠加层的垃圾帧。如果一个帧完成的时间超过约16毫秒(对于60 FPS设备),则该帧被认为是过时的。为了达到60 FPS(每秒帧)的帧渲染速率,每个帧必须在约16 ms或更短的时间内渲染。错过此目标时,您可能会遇到UI混乱或掉帧的情况

Render Frames

当一个Flutter应用或者Flutter Engine启动时,它会启动(或者从池中选择)另外三个线程,这些线程有些时候会有重合的工作点,但是通常,它们被称为UI线程GPU线程IO线程。UI、GPU之间的工作流程如下:

image

为了生成一帧,Flutter engine首先装备了vsync锁存器,一个vsync的事件将会指示Flutter engine开始一些工作并最终绘制出新的帧呈现在屏幕上,vsync事件的生成频率会根据硬件平台的刷新率决定。

vsync首先会唤醒UI线程,UI线程的工作是将你代码中编写的Widget树转化为要渲染的RenderTree,Flutter中有三颗树的概念WidgetTreeElementTreeRenderTree,dart文件中的Widget树并不是最终参与绘制的,而只是方便开发者编写页面的一个配置。比如,我们指定这里有一个纵向列表Column,列表里有三个并列Text,然后Flutter会根据相应语义在对应位置生成对应Element,这才是真正意义上的Flutter UI组件,也是显示到屏幕上的元素。

组件树对应到屏幕上还要经过一层渲染树(RenderObject)的转化,RenderObject是实际的渲染对象它负责布局测量以及绘制操作,这样做的目的是为了更好的应对上层UI的频繁变化,尽可能地去比较更新,修改配置而不是直接创建下层树,因为RenderObject树的创建开销比较大,所以Widget重新创建,ElementTree和RenderTree并不会完全重新创建,而是会复用一些节点,提升性能。UI线程工作到生成RenderTree的过程叫做渲染树

一旦创建了渲染树,GPU线程就会被唤醒,这个线程的工作是将渲染树的信息转换到GPU的命令缓冲区,然后在同一线程将数据提交给GPU执行

示例

模拟一个组件耗时操作


class PageOne extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Container(
      color: Colors.tealAccent,
      child: Center(
        child: ListView(
          children: [
            for(var i=0;i<100000;i++) _buildItemWidget(i),
          ],
        ),
      ),
    );
  }

  Widget _buildItemWidget(int i) {
    var line = lines[i % lines.length];
    return Padding(
        padding: EdgeInsets.symmetric(vertical: 12,horizontal: 18),
      child: Row(
        children: [
          Container(
            color: Colors.black,
            child: SizedBox(
              width: 30,
              height: 30,
              child: Center(
                child: Text(
                  line.substring(0,1),
                  style: TextStyle(color: Colors.white),
                ),
              ),
            ),
          ),
          SizedBox(width: 10,),
          Expanded(child: Text(
            line,
            softWrap: false,
          ))
        ],
      ),
    );
  }
}

可以看到,在ListView的children填充时,没有复用布局,模拟了一个重复创建十万条子child的情况,页面第一次加载时会看到明显卡顿,timeline显示如下,一条明显的红线,就是掉帧发生的位置。

image

点击Jank发生的位置,可以看到Timeline Events对应的事件被选中,下方有各个方法执行的耗时时间,可以看到_buildItemWidget方法耗时26.81ms发生掉帧

模拟一个方法耗时


class PageTwo extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Container(
      color: Colors.greenAccent,
      child: Center(
        child: Text(
          '第2页 ' + _fibonacci(30).toString(),
          style: TextStyle(color: Colors.black, fontSize: 20.0),
        ),
      ),
    );
  }

  static int _fibonacci(int i) {
    if(i <= 1) return i;
    return _fibonacci(i - 1) + _fibonacci(i - 2);
  }
}

image

组建初始化时,执行一个斐波那契函数递归调用,时间复杂度为O(2n),传入参数30,即函数运行230次运算,初始化页面可看到明显卡顿,定位耗时方法同上。

示例代码已上传至github

拓展

这里给大家推荐一个小工具fps_monitor,贝壳同学开源的检测页面流畅度的小工具,可以更直观和量化的评估页面流畅度,页面大概长这样

image

最大耗时平均耗时可以直观的观测页面优化前后对比效果。

页面流畅度划为了四个级别:流畅(蓝色)良好(黄色)轻微卡顿(粉色)卡顿(红色),将 FPS 折算成一帧所消耗的时间,不同级别采用不一样的颜色,统计不同级别出现的次数

具体可以跳转链接:
https://juejin.cn/post/6947911434424549384

总结

性能优化在任何平台任何语言上都是永恒不变的话题,理解性能优化原理,提升观察的敏锐性对一个开发者至关重要。利用Flutter提供的插件和性能分析工具,能够帮助我们快速的定位到问题代码,提升开发效率,Flutter Inspector可以在写代码阶段提升页面编码质量,Timeline可以在运行阶段发现哪个页面掉帧严重,重点分析。

参考链接

https://medium.com/flutter/profiling-flutter-applications-using-the-timeline-a1a434964af3

https://cloud.tencent.com/developer/article/1614400

https://juejin.cn/post/6940134891606507534

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

推荐阅读更多精彩内容

  • 1.前言 flutter_deer这个项目开源也近一年了,目前收获了3100+的star,这无疑是对这个项目的最大...
    唯鹿_weilu阅读 1,940评论 3 10
  • 在flutter的开发和工作中,因为工作内容的要求越来越高,加上一位优秀的同事,自己也对自己的写的代码除了规范的要...
    iOS弗森科阅读 1,884评论 0 6
  • 👏欢迎前往本人的GitHub查看更多内容。点击前往GitHub 在flutter的开发和工作中,因为工作内容的要求...
    iOS超级洋阅读 678评论 0 0
  • 表情是什么,我认为表情就是表现出来的情绪。表情可以传达很多信息。高兴了当然就笑了,难过就哭了。两者是相互影响密不可...
    Persistenc_6aea阅读 124,473评论 2 7
  • 16宿命:用概率思维提高你的胜算 以前的我是风险厌恶者,不喜欢去冒险,但是人生放弃了冒险,也就放弃了无数的可能。 ...
    yichen大刀阅读 6,041评论 0 4