Flutter中Widget的生命周期
Widget 的简介
先看一下Widget
. 从 Flutter的注释来看, 官方对Widget
的解释是:Describes the configuration for an [Element]
, 对于Flutter Framework来说, 并且是一个immutable description
, 在后续会关联到Element
.
Widget - Element - RenderObject 三棵树的内容后面会整理
有三个重点需要:
- Widget 是一个不可变的配置信息
- Widget 会关联
Element
类型 - 抽象类, 实际在开发时会继承子类:
StatefulWidget
,StatelessWidget
. StatefulWidget
的可变状态完全依赖于关联的State
abstract class Widget extends DiagnosticableTree {
const Widget({ this.key });
final Key key;
@protected
Element createElement();
...
/// Whether the `newWidget` can be used to update an [Element] that currently
/// has the `oldWidget` as its configuration.
static bool canUpdate(Widget oldWidget, Widget newWidget) {
return oldWidget.runtimeType == newWidget.runtimeType
&& oldWidget.key == newWidget.key;
}
}
StatelessWidget
的生命周期
StatelessWidget
作为无状态的Widget
, 生命周期比较简单:
const StatelessWidget({ Key? key }) : super(key: key);
构造函数StatelessElement createElement() => StatelessElement(this);
Widget关联的Element,Widget build(BuildContext context);
说明The framework calls this method when this widget is inserted into the tree
//Element 内部会只有 widget
/// An [Element] that uses a [StatelessWidget] as its configuration.
class StatelessElement extends ComponentElement {
/// Creates an element that uses the given widget as its configuration.
StatelessElement(StatelessWidget widget) : super(widget);
/// 使用
@override
StatelessWidget get widget => super.widget as StatelessWidget;
@override
Widget build() => widget.build(this);
@override
void update(StatelessWidget newWidget) {
super.update(newWidget);
_dirty = true;
rebuild();
}
}
StatefulWidget
的生命周期
StatefulWidget
是有状态的Widget
, 与之关联的State
的生命周期如下:
- widget:
const StatefulWidget({ Key? key }) : super(key: key);
- widget:
StatefulElement createElement() => StatefulElement(this);
- widget:
State createState()
- state:
constructor
- state:
initState
- state:
didChangeDependencies
(若InheritedWidget.rebuild
=>child.didChangeDependencies
) - state:
Widget build(BuildContext context);
(setState()
=>didUpdateWidget
=>build
) - state: 将
Widget - Element
加入tree
中 - state:
deactivate
- state:
dispose
- 将
Element
移除true
- InheritedWidget 方法 rebuild 时, 会调用childNode 中的
didChangeDependencies
- setState() 方法时, 会调用
didUpdateWidget
方法, 然后调用build
以上过程中几个关键的内容:
- 在
StatefulWidget
在调用createElement()
方法时, 关键的内容:
// StatefulWidget 中在 Widget被 mount 到 tree 时调用
StatefulElement createElement() {
// Element 的构造函数
return StatefulElement(this);
}
/// Creates an element that uses the given widget as its configuration.
/// Element构造函数中
/// 1. 会主动调用 widget.createState()方法, 并创建 _state!!!
/// 2. 然后, 在state中持有 element 和 widget
StatefulElement(StatefulWidget widget): _state = widget.createState(), super(widget) {
...
state._element = this;
state._widget = widget;
}
-
在
createState()
时, 可能会调用多次.For example, if the widget is inserted into the tree in multiple locations, the framework will create a separate [State] object for each location
-
SetState
的实现与dirty Element
的标记与更新:/// State 中的方法 void setState(VoidCallback fn) { // 先调用 fn() ... final Object? result = fn() as dynamic; _element!.markNeedsBuild(); } /// Element 中的方法 /// Marks the element as dirty and adds it to the global list of widgets to /// rebuild in the next frame. /// /// Since it is inefficient to build an element twice in one frame, /// applications and widgets should be structured so as to only mark /// widgets dirty during event handlers before the frame begins, not during /// the build itself. void markNeedsBuild() { ... _dirty = true; // 仅仅标记 Element, 会加入到全局 dirty list 中, 下一帧循环时, 会更新 owner!.scheduleBuildFor(this); // owner 是在调用 mount 时, 传入的parent.owner设置的 } // BuildOwner 类 -- Framework 中的管理类 /// Adds an element to the dirty elements list so that it will be rebuilt /// when [WidgetsBinding.drawFrame] calls [buildScope]. void scheduleBuildFor(Element element) { ... _dirtyElements.add(element); // 将 element 添加到全局的 dirty list 中 element._inDirtyList = true; }
转载自:https://juejin.cn/post/7051217146688307214