Flutter调优--深入探究MediaQuery引起界面Rebuild的原因及解决办法
前言
我们可以通过MediaQuery.of(context)
方法获取到一些设备和系统的相关信息,比如状态栏的高度、当前是否是黑暗模式等等,使用起来相当方便,但是也要注意可能引起的页面rebuild问题。本文会介我们可以通过MediaQuery.of(context)
方法获取到一些设备和系统的相关信息,比如状态栏的高度、当前是否是黑暗模式等等,使用起来相当方便,但是也要注意可能引起的页面rebuild问题。本文会介绍一个典型的例子,并深入源码来探讨引起rebuild的原因,最后介绍避免rebuild的几个办法。
绍一个典型的例子,并深入源码来探讨引起rebuild的原因,最后介绍避免rebuild的几个办法。
典型例子
以快递App中的查快递场景举例,首页用MediaQuery.of(context).padding.top
获取了状态栏高度,用户点击“查快递”按钮会跳转到查快递界面,在查快递界面,用户输入单号可进行查询操作。
当首页的build方法被调用时,会输出我们提前加好的日志。我们发现,当查快递界面的键盘弹出时,首页的build方法被调用了多次:
主界面的build代码如下:
源码探究
既然是因为主界面在build方法里使用了MediaQuery.of(context)
,从而导致当键盘弹出/隐藏时进行rebuild操作,那么就先来看下MediaQuery
类。
MediaQuery
其继承自InheritedWidget
,自身并没有重写createElement
方法,从flutter三棵树的角度讲,对应的Element
即为InheritedElement
。有两个属性,data和child,我们可以从data中获取一些设备/系统相关的属性。
另外还有两个比较重要的方法:
fromWindow(key : Key, child : Widget)
此方法直接返回_MediaQueryFromWindow
对象,后面会详细介绍。
of(context : BuildContext)
方法里调用了dependOnInheritedWidgetOfExactType,接下来我们详细分析下背后的调用流程。
MediaQuery.of(context) 调用流程
入参是context
,本例中的主界面是StatelessWidget
,那么这里的context
便是StatelessElement
。整体调用流程如下:
dependOnInheritedWidgetOfExactType
从_inheritedWidgets
列表中查询是否有MediaQuery
类型的InheritedElement
,从三棵树的角度讲,就是从当前节点一直向上查找,找到最近的MediaQuery
控件。如果找到,则调用dependOnInheritedElement
方法(一般情况下是一定能找到的,下面再详细介绍)。
dependOnInheritedElement
此方法负责将找到的InheritedElement
(也就是MediaQuery
对应的Element
)存起来,并且调用InheritedElement#updateDependencies
方法。
updateDependencies
setDependencies
最后两个方法很简单,其作用是将主页对应的StatelessElement
存储到了MediaQuery
对应的InheritedElement#_dependents
中。
研究完MediaQuery.of(context)
背后的原理,我们可以知道:通过调用of方法,主界面对应的Element
和MediaQuery
建立了绑定关系,MediaQuery
对应的InheritedElement
存储了主界面Element
的引用。
Rebuild起点
当介绍dependOnInheritedWidgetOfExactType
方法时,我们提道:从当前节点往父节点寻找,一般情况下是一定能找到的MediaQuery
控件的。这是因为在WidgetsApp
里会自动给我们创建一个根MediaQuery
。
在main
方法里,无论使用CupertinoApp
还是MaterialApp
,最后都会在内部创建WidgetsApp
。我们直接看_WidgetsAppState#build
方法里的一个代码片段:
会首先检查widget.useInheritedMediaQuery
,这个属性默认为false
。如果你创建MaterialApp
/CupertinoApp
时,没有设置useInheritedMediaQuery
属性,或者设置了这个属性为null,但找不到MediaQueryData
,那么这里就会调用MediaQuery.fromWindow
方法。
上面介绍MediaQuery#fromWindow
时,我们知道它会创建_MediaQueryFromWindow
控件。
_MediaQueryFromWindow
的代码不是很多,把和本文相关的代码全部贴出来了,大家可以自己看下,代码如上图所示。
build
方法里创建了MediaQuery
控件,并实现了didChangeMetrics
方法,当手机发生旋转、键盘弹出/隐藏时就会调用此方法,didChangeMetrics
内部又调用了setSate
,从而导致build
方法被重新调用。
通过flutter三颗树的原理我们可以知道,上述所说的“build方法被重新调用”涉及到MediaQueryFromWindow
对应的Element
的updateChild
方法,简单看下updateChild
的内部处理规则:
对MediaQueryFromWindow而言,每次都会创建新的MediaQuery Widget,根据Element#updateChild源码(不是本文讨论重点,不再详细分析其源码)得知,最终会调用MediaQuery对应的Element的update方法。
经过一系列的跳转过后,最终会调用到下面的两个核心方法:
上面介绍的MediaQuery.of(context)
方法最终会把入参Context
放到_dependents
变量里,而这里会遍历这个map
,调用每一个Context
的didChangeDependecies
方法,didChangeDependecies
会将此Context
置为dirty状态,下一帧来临时会被重新绘制,并调用此Context
的build
方法。
所以,破案了,当键盘弹起/隐藏时快递主页会被rebuild的原因找到了!
整体的rebuild调用流程如下,感兴趣的可以结合这个调用流程图去看源码:
避免rebuild的办法
研究过源码后,解决方案就变的很简单。
- 自定义
useInheritedMediaQuery
属性为true,并在最外面包一层MediaQuery
,让WidgetsApp
创建时使用MediaQuery
,而不去使用监听了application尺寸变化的_MediaQueryFromWindow
控件。
- 避免在页面中使用
MediaQuery.of(context)
方法,可以使用对应的替代方法,比如本例可以采用下面的代码进行替代,注意单位的转换。
- 如果必须要使用
MediaQuery.of(context)
方法,可以使用Builder
控件包裹下,of方法的入参传入此Builder
的context
即可,这样被rebuild仅是Builder
控件包裹下的widget子树。
总结
app界面逐渐复杂时,我们不得不考虑去优化界面性能。本文中介绍的例子在开发中是很常见的,如果不了解MediaQuery.of的机制,可能会引起大量使用此方法的界面发生重绘操作,造成页面卡顿、帧率下降。我们详细分析了背后的源码逻辑,介绍了解决办法,希望能给大家的调优工作提供些许帮助。
作者:京东物流 沈明亮
来源:京东云开发者社区
Flutter调优--深入探究MediaQuery引起界面Rebuild的原因及解决办法的更多相关文章
- qt designer启动后不显示界面问题的原因与解决办法
Qt 5.6.1无论是在vs里双击ui文件还是直接启动designer.exe都一直无法显示界面,但任务管理器中可以看到该进程是存在的.前几天还正常的,但昨天加了一块NVIDIA的显卡(机器自带核显) ...
- Qt拖拽界面 (*.ui) 缩放问题及解决办法
问题 使用Qt Designer 设计的界面,在缩放的时候不能随着主窗口一起缩放. 解决办法 之前遇到这个问题的时候,都是直接重写resizeEvent接口来实现的,在自动生成的Ui_Widget或U ...
- Qt拖拽界面 (*.ui) 缩放问题及解决办法(在最顶层放一个Layout)
问题 使用Qt Designer 设计的界面,在缩放的时候不能随着主窗口一起缩放. 解决办法 之前遇到这个问题的时候,都是直接重写resizeEvent接口来实现的,在自动生成的Ui_Widget或U ...
- ctrl+alt+F1~6进入不了字符界面,黑屏的解决办法
ubuntu系统,我是ubuntu14.04 本来想装cuda,需要在字符界面下装,奈何按ctrl+alt+F1就黑屏了,按ctrl+alt+F7又可以正常回到图形界面,网上查了很多,有的方法也试过, ...
- 分享一个android debug模式,出现 waiting for debugger把界面卡住,取巧的解决办法
使用android studio开发程序时,有时会出现 waiting for debugger 卡住界面,软件无法正常debug运行的情况,很多网友分享了一些解决办法,比如: 1 打开cmd进入命令 ...
- cloudera manager的7180 web界面访问不了的解决办法(图文详解)
说在前面的话 我的机器是总共4台,分别为ubuntucmbigdata1.ubuntucmbigdata2.ubuntucmbigdata3和ubuntucmbigdata4.(注意啦,以下是针对Ub ...
- 安装Windows 10后PDF补丁丁等程序界面变得模糊的解决办法
对于使用高分辨率屏幕且屏幕缩放比例在 100%以上的用户,升级到 Windows 10 后将发现许多程序的界面,例如QQ.电脑管家.Windows本身的服务管理程序等等,都变得非常模糊,<PDF ...
- 微信SDK登录无法调起,微信SDK无法接收回调的几种解决办法
今天有位同事请求帮忙调试微信登录问题,他遇到了以下2个问题,所以,写篇日志备忘,如果有其它朋友遇到此类问题,都可以照此解决! 平时在开发中,有些开发者经常会遇到微信登录SDK登录时,无法调起微信客户端 ...
- activemq无法启动且后台管理界面进不去的解决办法
从官网下载了一个最新的activemq,目前最新版本是5.14.5 我下载的是windows版本,通过执行%activemq home%/bin/win64/InstallService.bat,可以 ...
- 【Spark深入学习 -14】Spark应用经验与程序调优
----本节内容------- 1.遗留问题解答 2.Spark调优初体验 2.1 利用WebUI分析程序瓶颈 2.2 设置合适的资源 2.3 调整任务的并发度 2.4 修改存储格式 3.Spark调 ...
随机推荐
- 如何针对海外不同地区进行音视频自动化测试?丨Dev for Dev 专栏
近年来由于全球性的新冠疫情,世界各地对实时音视频的需求猛增.不同国家和地区由于经济发展.国家政策等原因,网络环境有很大不同,如果要做好音视频体验,就需要分地域进行音视频指标测试.但是不论是外包,还是云 ...
- 工良出品:包教会,Hadoop、Hive 搭建部署简易教程
目录 导读 Hadoop.Hive 是什么 运行环境 Java 环境 Mysql 下载 Hadoop.Hive 和 驱动 安装 Hadoop core-site.xml hdfs-site.xml m ...
- GaussDB(DWS)运维:导致SQL执行不下推的改写方案
摘要:本文就针对因USING子句的书写方式可能导致MERGE INTO语句的执行不下推的场景,对USING子句的SQL语句进行改写一遍,整个SQL语句可以下推. 本文分享自华为云社区<Gauss ...
- 动手造轮子自己实现人工智能神经网络(ANN),解决鸢尾花分类问题Golang1.18实现
人工智能神经网络( Artificial Neural Network,又称为ANN)是一种由人工神经元组成的网络结构,神经网络结构是所有机器学习的基本结构,换句话说,无论是深度学习还是强化学习都是基 ...
- webgl 系列 —— 着色器语言
其他章节请看: webgl 系列 着色器语言 本篇开始学习着色器语言 -- GLSL全称是 Graphics Library Shader Language (图形库着色器语言) GLSL 是一门独立 ...
- ACM-CodeForces-#685(Div.2)
好久没见过CF有这么水的contest了,蒟蒻赶紧找找自信 A. Subtract or Divide #include<iostream> using namespace std; in ...
- [Linux]Linux发展历程
古人云,知其然知其所以然.马哲思想指导着我们,任何事物.问题,离不开:为什么(Why,事物从哪里来?).是什么(What,事物的定位?).怎么做(How,到哪里去?)的哲学3问. 继上个月算是相对彻底 ...
- Java线程池浅析
1. 什么是线程池?我们为什么需要线程池? 线程池即可以存放线程的容器,若干个可执行现成在"容器"中等待被调度. 我们都知道,线程的生命周期中有以下状态:新建状态(New).就绪状 ...
- 活动预告 | Jax Diffusers 社区冲刺线上分享(还有北京线下活动)
我们的 Jax Diffuser 社区冲刺活动已经截止报名,全球有 200 多名参赛选手成功组成了约 70 支队伍共同参赛. 为了帮助参赛者更好的完成自己的项目,也为了与更多社区成员们分享扩散模型和生 ...
- 【贪心算法】NO134 加油站
134. 加油站 在一条环路上有 n 个加油站,其中第 i 个加油站有汽油 gas[i] 升. 你有一辆油箱容量无限的的汽车,从第 i 个加油站开往第 i+1 个加油站需要消耗汽油 cost[i] 升 ...