Widget基础系列 - StatefulWidget

在Flutter中,Widget可以说是第一基础概念。Widget是对用户界面的不可变描述,可被膨化为管理底层渲染树的Element。

理解Widget原理是掌握Flutter编程至关重要的一步,本系列主要介绍Widget的基础知识,本文是第二篇:

在Flutter框架中,Widget处于中心位置。Widgets是对用户界面的不可变描述,可被膨胀为管理底层渲染树的elements。

理解Widget原理是掌握Flutter编程至关重要的一步,本系列主要介绍Widget的基础知识,本文是第二篇:

  • StatelessWidget
  • StatefulWidget
  • InheritedWidget
  • Key

StatefulWidget

本篇介绍有状态的widget -- 与无状态widget的区别, State对象的工作原理等。

看过本系列第一篇文章的,应该对无状态widget有所了解了,如果还没有看,或者还不了解无状态widget,建议先看第一篇。

我们知道,Element表示屏幕上的实际内容,而widget是element的不可变配置或蓝本。你肯定会有这样的疑问,我们的app不可能只是展示静态不变的内容,如何处理可变数据呢?如何跟踪数据并刷新UI你?这就需要用到StatefulWidget了。它除了提供不可变的配置信息之外,还提供了一个可随时间变化以及触发UI刷新的state对象。

我们通过一段简单的代码来看一下它是如何工作的。

class ItemCount extends StatelessWidget {
  final String name;
  final int count;

  ItemCount({this.name, this.count});

  @override
  Widget build(BuildContext context) {
    return Text('$name: $count');
  }
}

这是一个非常简单的无状态widget,构造函数接受两个参数:name和count,并构建了一个文本widget将它们显示出来。现在,我们希望count可以变化。在无状态widget中,我们无法修改任何东西,对吧?countfinal类型的。

因此,我们将它转变为一个有状态的widget:

class ItemCounter extends StatefulWidget {
  final String name;

  ItemCounter({this.name});

  @override
  _ItemCountState createState() => _ItemCountState();
}

class _ItemCountState extends State<ItemCounter> {
  int count = 0;

  @override
  Widget build(BuildContext context) {
    return Text('${widget.name}: $count');
  }
}

现在有了两个类,一个widget类和一个state类。Widget类有两个职责:持有不可变的name,以及创建state对象。而state对象呢,持有count,可以注意到,count不再是final的,而是可变的,并且子widget此时由state对象负责创建。Text将widget中不可变的name和state中可变的count组合之后显示出来。

要了解这是如何工作的,让我们来看一下widget和element树。

stateless_widget_element_tree

看过第一篇文章的应该知道,element树表示屏幕上的实际显示内容,widget仅仅是element的蓝本。对于无状态widget来说,显示过程是非常直接的。你给Flutter一个无状态widget,Flutter向这个widget请求一个element,并将其挂载到element树上。如果这个无状态widget构建了子widget,则同样向它们请求element对象,并挂载到element树上。

对于有状态widget来说,有一个额外的步骤。同无状态widget一样,首先从widget开始,Flutter请求有状态widget创建一个element对象,有状态widget返回一个StatefulElement对象。然后,这个StatefulElement对象向widget对象请求一个state对象,这就是createState方法的用途。此方法返回一个新的state对象,element对象会持有这个对象。

widget_element_state

开始构建子widget,StatefulElement调用state对象的build方法。再看一下前面的代码,为了构建文本,我们需要widget中的name属性和state对象的count属性。因为state对象维护了一个对widget对象的引用,因此它可以访问这两个值以构建文本。就是这样:

item_counter_widget_element_tree

Text是无状态的,因此它创建了一个StatelessElement,并挂载到element树上。从技术上说,Text还有一些自己的子widget,以提供辅助选项、渲染文本等功能。不过,对于这个简单的例子来说,我们不再深入,keep it simple。

一切都已创建,element树可以开始工作了。不过,再看一下state对象,此时还没有对象更新state,没有什么改变count属性。

class _ItemCountState extends State<ItemCounter> {
  int count = 0;

  @override
  Widget build(BuildContext context) {
    return Text('${widget.name}: $count');
  }
}

如果放入一个GestureDetector,就可以通过state对象的setState方法触发更新了。

class _ItemCountState extends State<ItemCounter> {
  int count = 0;

  @override
  Widget build(BuildContext context) {
    return GestureDetector(
      onTap: () {
        setState(() {
          count++;
        });
      },
      child: Text('${widget.name}: $count'),
    );
  }
}

setState是更新属性并刷新UI的一种方式,向它提供一个更新属性的函数,state对象运行此函数并刷新UI。

item_counter_widget_element_tree_setstate

看一下图表,当setState运行时,count加1,此处有一个关键点,state对象标记element为dirty,表示下一帧时需要重建子节点。当下一帧时,像之前一样,StatefulElement调用state对象的build方法重建子节点,输出一个新的Text以显示新的count值。酷的地方是,由于更新前后widget的类型一样 -- 都是TextStatelessElement依然保留在原处,仅更新widget引用到新的widget而已。

这是state对象持有可变数据,以及数据更新时重建子widget的基本例子。

对于state对象来说,还有很重要的一点 -- 它们的生命周期比较长。只要更新前后的widget类型不变(准确地说,还需要Key值相等,不过本文不涉及Key,可以暂时忽略,详见第四篇文章),当widget对象被新的对象替换时,state对象依然会附着于element树上。举例来说,当ItemCounter对象由于树上层的变化而重建时(name变了),原来的ItemCounter被移去,不过由于新的对象类型相同,StatefulElement和state对象依然保留在原处。

item_counter_widget_element_tree_update_name

它们从widget的更新中存活下来,仅标记自己为dirty以重建子节点。然后,state对象的build方法给出一个显示其count值的新Text,但name换成了新widget中的值。旧的Text对象被移去,新的被挂载上去,而Text对应的element对象依然保留在原处。这就是当创建state对象的widget被替换时状态还能保持的原因。就像热更新,向设备推送新的代码而不改变应用的状态。

State对象的生命周期与Widget不同,这一点必须谨记在心。

在这儿,我们使用新的属性构建新的widget,但是state对象不变。State类还有一个方法,如果state对象希望知道widget对象何时被替换,可以重写didUpdateWidget方法。例如,对于上面的ItemCounter,有如下代码:

item_counter_app

当我们点击“Change Name”按钮切换名称时,正如前面介绍的那样,count值并不会改变。如果我们希望当widget的name改变时将count值清空,该如何做呢?

class _ItemCountState extends State<ItemCounter> {
  int count = 0;

  @override
  Widget build(BuildContext context) {
    return GestureDetector(
      onTap: () {
        setState(() {
          count++;
        });
      },
      child: Text('${widget.name}: $count'),
    );
  }

  @override
  void didUpdateWidget(ItemCounter oldWidget) {
    if (oldWidget.name != widget.name) {
        count = 0;
    }
    super.didUpdateWidget(oldWidget);
  }
}

Flutter框架在调用didUpdateWidget方法之后会调用build方法,因此在didUpdateWidget方法中调用setState是多余的。

我们可以看到,StatefulWidget使我们可以很方便地跟踪数据的变化并刷新UI。不过,随着对Flutter的运用愈发熟练,我们会发现越来越少需要自己写StatefulWidget。一个原因是,很多常用的功能已经实现过了。例如,如果有一个数据流,我们需要一个当数据流输出新数据时能自动刷新的StatefulWidget,OK,Flutter框架中有一个StreamBuilder

StreamBuilder(
  stream: _myStreamOfStrings,
  builder: (context, snapshot) {
    Text(snapshot.data ?? 'loading...');
  }
)

另一个原因是,如果嵌套了很多StatefulWidget,通过这些widget的build和构造方法传递数据是非常笨重的。

stateful_widgets

幸运的是,在Flutter还有一种类型的widget,使我们可以轻松访问树中上层的数据,哪怕是隔了100层也没关系。这就是InheritedWidget,下篇文章会介绍它。

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

推荐阅读更多精彩内容

  • 首先,在Flutter中几乎所有的对象都是一个Widget。跟原生开发中的“控件”不同,Flutter中的Widg...
    沉江小鱼阅读 1,617评论 0 2
  • 原文在此,此处只为学习 Widget与ElementWidget主要接口Stateless WidgetState...
    lltree阅读 4,510评论 0 1
  • 在Flutter中,Widget可以说是第一基础概念。Widget是对用户界面的不可变描述,可被膨化为管理底层渲染...
    光明自在阅读 1,488评论 0 7
  • 前言 上一篇我们简单地了解了 Dart 语言,接着我们就开始学习 Flutter 的基础 Widget 吧。 1....
    南小夕阅读 2,568评论 0 8
  • 1 Widget只是UI元素的一个配置数据,并且一个Widget可以对应多个ElementWidget实际上就是E...
    你飞跃俊杰阅读 725评论 0 2