当前位置: 首页 > wzjs >正文

湘潭网站建设 真好磐石网络360推广登陆

湘潭网站建设 真好磐石网络,360推广登陆,建设部中国建设工程信息网,成都眉山网站建设Flutter 的 Widget Key 提议大调整?深入聊一聊 Key 的作用 在 Flutter 里,Key 对象存在的目的主要是区分和维持 Widget 的状态,它是控件在渲染树里的「复用」标识之一,这一点在之前的《深入 Flutter 和 Compose 在 UI 渲染刷新时…

Flutter 的 Widget Key 提议大调整?深入聊一聊 Key 的作用

在 Flutter 里,Key 对象存在的目的主要是区分和维持 Widget 的状态,它是控件在渲染树里的「复用」标识之一,这一点在之前的《深入 Flutter 和 Compose 在 UI 渲染刷新时 Diff 实现对比》 聊到过,可以说 Key 的存在关乎了 Flutter 的性能,因为它的作用就是提高 Element Tree 的复用效率,例如减少匹配阶段所需的 Widget 比较次数。

另外通过 Key 还可以提高如 AnimatedListListView 里重新排序时对应 Item widget 的效率,通过将 Key 分配给 Item ,Flutter 可以更有效地识别何时添加、删除或更新列表并执行动画,在这个时候, Key 可以确保每个 Item 即使在对列表进行排序时也保持其状态。

大多数情况下,无状态的 Widget 是不需要 Key,而默认情况下,我们在不主动配置 Key 的时候,它会是 null :

也就是在没有 Key 的情况下,framewok 一般只判断 runtimeType 去决定是否「复用」,举个很老的官方例子,如下图片代码里的 StatelessColorfulTile 所示,它是一个无状态的 StatelessWidget ,显示了一个随机颜色的 200x200 大小的正方形,通过点击右下角按键,每次调整两个方块的位置,可以看到方块可以正常切换:

因为此时没有 Key ,在 Element Tree 只需要判断 runtimeType ,明显此时 Element 符合复用条件,而代码里又是直接使用 StatelessColorfulTile 的 Widget 实例对象进行 tiles.insert(1, tiles.removeAt(0)) ,所以在 Widget 切换位置之后,Element 和 RenderObject 只需要 update 一下新位置 Widget 实例的颜色即可:

但是,如果我们修改为 StatefulWidget ,此时我们再点击右下角按键,可以看到此时颜色方块不会切换了:

因为此时颜色 color 被保存在 State 下,在 Widget 切换位置之后,因为 runtimeType 符合条件,所以 Element 复用,但是颜色被保存在 State 下,State 又是保存在 Element 里,从而导致颜色并没有按照需求被更新切换:

但是,如果这时候我们给两个 StatefulWidget 添加上 Key ,就可以看到它们可以被切换了,因为 canUpdate 判断条件会增加 Key 判断:

也就是,在有了 Key 之后,新 Widget 的 key 就可以在老 Element 列表里进行匹配,从而更新 Element 的位置并刷新 RenderObject,两个 Element 在状态保留的情况下,被 Tree 里调换了位置进行更新,从而实现了切换的效果:

所以,从这个简单的例子,可以直观看到 Key 在有状态的情况下能够发挥的作用,当然,目前在 Flutter 里的 Key 类型很丰富,但是大致可以简单分为两类: Local Keys 和 Global Keys

顾名思义就是它的作用范围,举个例子,如果我们给 StatelessColorfulTile 增加了一个 Padding ,再点击切换按键,可以看到此时点击后 Element 一直被重构:

因为此时在 Row 里面,此时处于“一级”位置 children 是两个 Padding,而 Padding 没有 Key,所以它在 runtimeType 条件的情况下,是直接被复用:

而对于 StatelessColorfulTile 而言,它处于 Padding 之下,Padding 不是一个 Multi Child 的控件,所以在 canUpdate 为 false 的时候,Flutter 内部会认为它需要被重新创建:

从这里我们就可以很直观体验到 Local Keys 这个概念:它只作用于标识同一父 Widget 中的 Widget,不能用于识别其父 Widget 之外的 Widget

同时,我们也可以是直观感受到:Multi Child 和 Single Child 的 Element 对于 Diff 更新时的策略差别

另外,我们还可以感受到 Widget 作为「配置文件」的存在,要知道,代码里我们操作的一直都是 tiles.insert(1, tiles.removeAt(0)); ,也就是 Widget 的实例化都的对象,虽然 Widget 实例没变,但是 Element 层面还是会根据情况「重新创建」对应的 Element ,由于颜色是在 State 里,所以也就会跟着 Element 重新随机变化。

最后如下图所示,对于 Local Keys 来说,左侧这样的写法是可以的,而右侧这样的写法是违规的:

所以,在 Widget 的 Key 注释里也有这样一句描述:通常情况下,作为另一个 widget 的唯一子项的 widget 不需要显式 Key

GlobalKey

那么,除开 Local Keys ,Flutter 里还有一个特殊的 GlobalKey,允许开发者在 Widget 树里去「唯一」标识 Widget,并提供 BuildContext(Element)/State 的全局访问:

这里的「唯一」更多体现在当前这一帧里的「唯一」。

比如前面的例子,我们只需要把对应的 Local Keys 换成 GlobalKey ,就可以看到,虽然 Key 所在的 StatelessColorfulTile 还是在 Padding 下的“二级” child ,但是现在点击切换时,它不会被「重新创建」导致颜色发生变化:

这是因为,虽然在 updateChild 的时候,逻辑依然会走到 inflateWidget 去创建 Element ,但是由于是 GlobalKey,所以会从全局保存的 Map 里获取到当前 GlobalKey 绑定的 Element ,从而 retake 复用:

从这里可以看出来, 如果 Element 在同一帧中移动或者删除,并且它具有 GlobalKey,那么它仍然可能被重新激活使用

所以 GlobalKey 不仅可以作为 Key 区分 Widget ,帧内还可以在 BuildOwner 里“全局”保持住 Element 、State 和关联 RenderObject 的“状态”,即使它出现移动或者删除。

同时,通过 GlobalKey ,我们也可以访问对应的 BuildContext 和 State 数据,甚至是直接给 MaterialApp 添加 GlobalKey 来操作导航:

那么 GlobalKey 这么好用,它又存在什么问题呢?其实在注释里已经有对应说明:

GlobalKey 在使用的过程中可能会出现需要重新设置 [Element] 父级的情况,而这个操作会触发对关联的 [State] 及其所有后代 [State.deactivate] 的调用,还会强制重建所有依赖于 [InheritedWidget] 的控件。

具体就体现在这下面两段代码:

  • _retakeInactiveElement 内可能会触发所有关联 State 的 deactivate
  • _activateWithParent 会触发 Element 的 activate ,从而通过 didChangeDependencies 强制重建所有依赖于 [InheritedWidget] 的控件

当然,GlobalKey 也有一些注意事项,例如:

使用 GlobalKey 不能频繁创建,通常应该是让它和 State 对象拥有类似的“生命长度”,因为新的 GlobalKey 会丢弃与旧 Key 关联的子树的状态,并为新键创建一个新的子树,频繁创建会导致状态丢失和性能损耗。

变更提议

前面我们主要介绍了 Key 的作用和分类下的职能,而本次 PR 提议的调整,则是在于打算简化 Local Keys 相关的实现上,可以看到在以往的实现里,关于 LocalKey 的实现有好几种类型,但是其中一些职能其实「相对重复」:

在 #159225 的 PR 里,将打算把 Key 对象切换到 Object ,从而“消灭”过往这些 Local Keys 的“重叠”,让 Key API 更加灵活:

另外,除了灵活和简化之外,针对目前存在的 Local Keys ,它和 Dart 的 Extension Types 不同,比如使用 ValueKey() 多多少少会有一点点点点点点 wrapper 成本,而如果这个提议合并后,大概会是如下所示的情况,或多或少对性能还是有那么一点点点点点点帮助:

事实上对于 LocalKey ,大多数人应该都只会使用到 ValueKey 居多。

当然,这个 PR 整体来说还是属于底层大调整,而目前看起来提议应该是暂时搁置了,不过就算推进落地,相信对于大多数上层 Flutter 开发者来说,应该也不会有明显的感知,毕竟大多数时候 Flutter 开发者对 Key 并不敏感:

所以,你是喜欢现在的 Local Keys 分类还是提议里的 Object ?

参考链接:

  • https://github.com/flutter/flutter/pull/159225
  • https://api.flutter.dev/flutter/foundation/Key-class.html
http://www.dtcms.com/wzjs/407199.html

相关文章:

  • 哪个网站做自考题目免费百度信息流优化
  • 在线甜品网站开发书谷歌seo招聘
  • 女生适合计算机哪个专业网络推广优化招聘
  • 做旅游网站的网站优化的方法与技巧
  • 广东网站建设引流推广营销
  • 找个做网站的 优帮云百度人工服务
  • wordpress右下角设置免费seo排名优化
  • 官方网站建设银行年利息是多少钱北京搜索排名优化
  • 有哪些网站可以做海报如何快速推广app
  • 营销型网站排友情下载网站
  • 网站怎么做关键词库cnzz
  • 建站公司哪个平台最好营销策略怎么写
  • 用dw如何做网站链接免费的黄冈网站有哪些平台
  • 网络营销相关理论电商网站怎样优化
  • 网站文字特效广东广州网点快速网站建设
  • 苏州网站建设选苏州梦易行网站排名seo教程
  • 中央廉政建设网站磁力猫torrent kitty
  • 企业宣传网站系统建设方案长沙官网seo技巧
  • 山西营销型网站建设互联网搜索引擎有哪些
  • 网站建设合同 域名商务软文写作300字
  • 网站建设师怎么在百度上发布信息
  • 京东网站建设现状搜索引擎网页
  • 广州力yang网站建设微信群发软件
  • 温州做网站多少钱app001推广平台
  • 阿里云网站全部清空怎么做创新营销方式有哪些
  • 自己怎么做直播网站吗seo排名规则
  • 苏州做手机网站广州seo优化排名公司
  • 旅游网站建设推广百度合伙人答题兼职赚钱
  • h5制作工具网站网络优化的流程
  • 做旅游网站的需求分析买外链有用吗