Flutter Widget源码解析及实战(一)

本文涉及的产品
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: Flutter Widget源码解析及实战(一)

Widget


在flutter中所有页面展示出来的元素都是由一个个的widget组成,与原生android开发不同的地方在于flutter中widget不仅仅表示UI元素,他也可以是一个完全和UI无关如GestureDetector,GestureDetector继承自StatelessWidget。


Widget的功能类似于原生android开发中的style文件,用来描述UI的样式,最终真正绘制在屏幕上的是Element。

@immutableabstract class Widget extends DiagnosticableTree { // DiagnosticableTree 诊断树主要作用是提供调试信息    const Widget({ this.key }); // key 
  // 最终整个绘制流程在Element中进行  @protected  Element createElement();
  @override  String toStringShort() {    return key == null ? '$runtimeType' : '$runtimeType-$key';  }
  // 提供一些调试用的信息  @override  void debugFillProperties(DiagnosticPropertiesBuilder properties) {    super.debugFillProperties(properties);    properties.defaultDiagnosticsTreeStyle = DiagnosticsTreeStyle.dense;  }
  // 这里通过判断key和runtimeType是否一致来判断是否可以更新UI  static bool canUpdate(Widget oldWidget, Widget newWidget) {    return oldWidget.runtimeType == newWidget.runtimeType        && oldWidget.key == newWidget.key;  }}


StatelessWidget


无状态的widget一般用于一些静态UI的绘制(例如:Text)或者提供与UI无关的功能(例如:GestureDetector用来管理手势相关的功能),源码如下:


abstract class StatelessWidget extends Widget {    const StatelessWidget({ Key key }) : super(key: key);    @override  StatelessElement createElement() => StatelessElement(this);    @protected  Widget build(BuildContext context);  }


StatelessWidget用于不需要维护状态的场景,它通常在build方法中通过嵌套其它Widget来构建UI,在构建过程中会递归的构建其嵌套的Widget,具体如下:


class TestLess extends StatelessWidget {  @override  Widget build(BuildContext context) {    return Container(      child: Text("test"),    );  }}

StatefulWidget


可变状态的小部件

abstract class StatefulWidget extends Widget {
  const StatefulWidget({ Key key }) : super(key: key);
  @override  StatefulElement createElement() => StatefulElement(this);
  @protected  State createState();}


与StatelessWidget不同的是StatefulWidget类中添加了一个新的接口createState(),一个StatefulWidget类会对应一个State类,State表示与其对应的StatefulWidget要维护的状态。下面是StatefulWidget的最佳实践:


  • 尽量将需要该表状态的widget防止在子节点,这样在改变整个渲染树的时候就只需要更新一个widget即可,如果将其防止在父节点那么将会导致当前节点的整个子节点的widget都会被重新渲染。
  • 尽量减少build方法中返回的widget的嵌套层级,理想情况下一个StatefulWidget仅仅只包含一个类型为RenderObjectWidget的子widget。例如:RichText,但显然这是不切实际的,但一个小部件越是接近这个理想,效率越高。
  • 如果子树没有更改,请缓存表示该子树的窗口小部件,并在每次使用时重新使用它。对于要重新使用的窗口小部件,要比创建新的(但配置相同的)窗口小部件更有效。将有状态部分分解为带有子参数的小部件是执行此操作的常用方法。
  • 尽可能使用`const`小部件。(这相当于缓存窗口小部件并重新使用它。)
  • 避免更改任何创建的子树的深度或更改子树中任何窗口小部件的类型。例如,不是返回包含在[IgnorePointer]中的子项或子项,而是始终将子窗口小部件包装在[IgnorePointer]中并控制[IgnorePointer.ignoring]属性。这是因为更改子树的深度需要重建,布局和绘制整个子树,而只更改属性将需要对渲染树进行尽可能少的更改(例如,在[IgnorePointer]的情况下,没有布局)或重绘是必要的)。
  • 如果由于某种原因必须更改深度,请考虑将子树的公共部分包装在具有[GlobalKey]的小部件中,该[GlobalKey]在有状态小部件的生命周期内保持一致。(如果没有其他小部件可以方便地分配密钥,[KeyedSubtree]小部件可能对此有用。)


下面是一个名为`YellowBird`的有状态小部件子类的框架。在这个例子中[State]没有实际状态。State通常表示为私人成员字段。此外,通常小部件有更多的构造函数参数,每个参数都应该为`final`类型。

class YellowBird extends StatefulWidget {   const YellowBird({ Key key }) : super(key: key);
   @override   _YellowBirdState createState() => _YellowBirdState(); }
 class _YellowBirdState extends State<YellowBird> {   @override   Widget build(BuildContext context) {     return Container(color: const Color(0xFFFFE306));   } }


下面的例子显示了更通用的小部件`Bird`,它可以被赋予一种颜色和一个子widget,并且它有一些内部状态,可以调用一个方法来改变它。

class Bird extends StatefulWidget {   const Bird({     Key key,     this.color = const Color(0xFFFFE306),     this.child,   }) : super(key: key);
   final Color color;   final Widget child;
   _BirdState createState() => _BirdState(); }
 class _BirdState extends State<Bird> {   double _size = 1.0;
   void grow() {     setState(() { _size += 0.1; });   }
   @override   Widget build(BuildContext context) {     return Container(       color: widget.color,       transform: Matrix4.diagonal3Values(_size, _size, 1.0),       child: widget.child,     );   } }

按照惯例,窗口小部件构造函数仅使用命名参数。可以使用[@required]将命名参数标记为必需。按照惯例,第一个参数是[key],最后一个参数是`child`,`children`。


StatefulWidget生命周期


image.png

State中有两个常用属性


  • widget  :表示与State实例相关联的widget实例
  • BuildContext:构建widget的上下文
  • initState:当Widget第一次插入到Widget树时会被调用,对于每一个State对象。framework将在创建的每个[State]对象调用此方法一次。重写此方法以执行初始化,该初始化取决于此对象插入树中的位置(即[context])或用于配置此对象的窗口小部件(即[widget])。如果[State]的[build]方法依赖于一个本身可以改变状态的对象,例如[ChangeNotifier]或[Stream],或者一个可以订阅接收通知的其他对象,那么一定要订阅并在[initState],[didUpdateWidget]和[dispose]中取消订阅:您不能使用此方法中的[BuildContext.inheritFromWidgetOfExactType]。但是,[didChangeDependencies]将在此方法之后立即调用,可以在那里使用[BuildContext.inheritFromWidgetOfExactType]。
  • didChangeDependencies:当State对象的依赖发生变化时会被调用,如果父Widget重建并请求树中的此位置更新以显示具有相同[runtimeType]和[Widget.key]的新Widget,则框架将更新此[State]对象的[widget]属性以引用新Widget然后使用上一个Widget作为参数调用此方法。覆写此方法可以在[widget]更改时进行响应(例如,开始隐式动画)。在调用[didUpdateWidget]之后,框架总是调用[build],这意味着对[didUpdateWidget]中的[setState]的任何调用都是多余的。
  • build:它主要是用于构建Widget子树
  • reassemble:此回调是专门为了开发调试而提供的,在热重载(hot reload)时会被调用。
  • didUpdateWidget:在widget重新构建时,framework会调用canUpdate来检测Widget树中同一位置的新旧节点,然后决定是否需要更新。
  • deactivate:当State对象从树中被移除时,会调用此回调。在一些场景下,Flutter framework会将State对象重新插到树中,如包含此State对象的子树在树的一个位置移动到另一个位置时(可以通过GlobalKey来实现)。如果移除后没有重新插入到树中则紧接着会调用dispose()方法。
  • dispose:当State对象从树中被永久移除时调用;通常在此回调中释放资源。
相关文章
|
8天前
|
数据采集 JSON 数据可视化
JSON数据解析实战:从嵌套结构到结构化表格
在信息爆炸的时代,从杂乱数据中提取精准知识图谱是数据侦探的挑战。本文以Google Scholar为例,解析嵌套JSON数据,提取文献信息并转换为结构化表格,通过Graphviz制作技术关系图谱,揭示文献间的隐秘联系。代码涵盖代理IP、请求头设置、JSON解析及可视化,提供完整实战案例。
JSON数据解析实战:从嵌套结构到结构化表格
|
11天前
|
索引
【Flutter 开发必备】AzListView 组件全解析,打造丝滑索引列表!
在 Flutter 开发中,AzListView 是实现字母索引分类列表的理想选择。它支持 A-Z 快速跳转、悬浮分组标题、自定义 UI 和高效性能,适用于通讯录、城市选择等场景。本文将详细解析 AzListView 的核心参数和实战示例,助你轻松实现流畅的索引列表。
33 7
|
15天前
|
数据可视化 测试技术 API
GraphQL开发工具选型指南:Apipost高效调试与文档生成实战解析
本文深入解析了GraphQL开发工具Apipost在高效调试与文档生成方面的优势,对比同类工具Apifox,突出其可视化界面、实时调试及自动化文档生成等特性。Apipost通过智能代码补全、错误提示等功能简化复杂Query编写,支持一键生成标准化文档,显著提升开发效率和团队协作效果,尤其适合中大型团队应对复杂业务场景。
|
2月前
|
运维 Shell 数据库
Python执行Shell命令并获取结果:深入解析与实战
通过以上内容,开发者可以在实际项目中灵活应用Python执行Shell命令,实现各种自动化任务,提高开发和运维效率。
80 20
|
2月前
|
供应链 搜索推荐 API
深度解析1688 API对电商的影响与实战应用
在全球电子商务迅猛发展的背景下,1688作为知名的B2B电商平台,为中小企业提供商品批发、分销、供应链管理等一站式服务,并通过开放的API接口,为开发者和电商企业提供数据资源和功能支持。本文将深入解析1688 API的功能(如商品搜索、详情、订单管理等)、应用场景(如商品展示、搜索优化、交易管理和用户行为分析)、收益分析(如流量增长、销售提升、库存优化和成本降低)及实际案例,帮助电商从业者提升运营效率和商业收益。
252 20
|
2月前
|
数据采集 XML API
深入解析BeautifulSoup:从sohu.com视频页面提取关键信息的实战技巧
深入解析BeautifulSoup:从sohu.com视频页面提取关键信息的实战技巧
|
3月前
|
安全 API 数据安全/隐私保护
速卖通AliExpress商品详情API接口深度解析与实战应用
速卖通(AliExpress)作为全球化电商的重要平台,提供了丰富的商品资源和便捷的购物体验。为了提升用户体验和优化商品管理,速卖通开放了API接口,其中商品详情API尤为关键。本文介绍如何获取API密钥、调用商品详情API接口,并处理API响应数据,帮助开发者和商家高效利用这些工具。通过合理规划API调用策略和确保合法合规使用,开发者可以更好地获取商品信息,优化管理和营销策略。
|
3月前
|
Dart 前端开发 IDE
鸿蒙Flutter实战:14-现有Flutter 项目支持鸿蒙 II
本文介绍了如何将现有 Flutter 项目适配鸿蒙系统,详细步骤包括安装 FVM、使用 FVM 安装 Flutter SDK、搭建开发环境、创建项目架构和壳工程等。
393 4
鸿蒙Flutter实战:14-现有Flutter 项目支持鸿蒙 II
|
3月前
|
自然语言处理 搜索推荐 数据安全/隐私保护
鸿蒙登录页面好看的样式设计-HarmonyOS应用开发实战与ArkTS代码解析【HarmonyOS 5.0(Next)】
鸿蒙登录页面设计展示了 HarmonyOS 5.0(Next)的未来美学理念,结合科技与艺术,为用户带来视觉盛宴。该页面使用 ArkTS 开发,支持个性化定制和无缝智能设备连接。代码解析涵盖了声明式 UI、状态管理、事件处理及路由导航等关键概念,帮助开发者快速上手 HarmonyOS 应用开发。通过这段代码,开发者可以了解如何构建交互式界面并实现跨设备协同工作,推动智能生态的发展。
233 10
鸿蒙登录页面好看的样式设计-HarmonyOS应用开发实战与ArkTS代码解析【HarmonyOS 5.0(Next)】
|
3月前
|
物联网 调度 vr&ar
鸿蒙HarmonyOS应用开发 |鸿蒙技术分享HarmonyOS Next 深度解析:分布式能力与跨设备协作实战
鸿蒙技术分享:HarmonyOS Next 深度解析 随着万物互联时代的到来,华为发布的 HarmonyOS Next 在技术架构和生态体验上实现了重大升级。本文从技术架构、生态优势和开发实践三方面深入探讨其特点,并通过跨设备笔记应用实战案例,展示其强大的分布式能力和多设备协作功能。核心亮点包括新一代微内核架构、统一开发语言 ArkTS 和多模态交互支持。开发者可借助 DevEco Studio 4.0 快速上手,体验高效、灵活的开发过程。 239个字符
271 13
鸿蒙HarmonyOS应用开发 |鸿蒙技术分享HarmonyOS Next 深度解析:分布式能力与跨设备协作实战

热门文章

最新文章

推荐镜像

更多