Flutter入门(十一)生命周期&渲染原理

前言

在做移动端开发时,不论是安卓的Activity还是iOS的UIViewController都有自己的生命周期,同理Flutter也有自己的生命周期,今天我们就通过以下几个方面来探索一下:

  • 1.生命周期的基本概念
  • 2.Widget的生命周期
  • 3.Flutter渲染原理

1.生命周期的基本概念

Flutter生命周期流程

1.1什么是生命周期

  • 1.说白了就是回调函数
  • 2.让你知道封装好的这个Widget处于什么样的状态

1.2生命周期的作用

  • 1.初始化数据
    1.1创建变量、常量
    1.2发送网络请求

  • 2.监听小部件的事件

  • 3.管理内存
    3.1销毁数据、销毁监听者
    3.2销毁Timer等等

2.Widget的生命周期

2.1StatelessWidget的生命周期

class MyWidget extends StatelessWidget {
  const MyWidget({Key? key}) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return Container();
  }
}
  • 1.构造方法
  • 2.build方法

对于StatelessWidget来说,生命周期只有构造和build过程。build是用来创建Widget的,在每次页面刷新时会调用build

2.2StatefulWidget的生命周期

class MyHomePage extends StatefulWidget {
  final String title;

  MyHomePage({Key? key, required this.title}) : super(key: key) {
    print('Widget构造方法来了');
  }

  @override
  State<MyHomePage> createState() {
    print('Widget的createState方法调用了');
    return _MyHomePageState();
  }
}

class _MyHomePageState extends State<MyHomePage> {
  _MyHomePageState() {
    print('State构造方法来了');
  }

  @override
  void initState() {
    print('State的initState来了');
    super.initState();
  }

  @override
  void didChangeDependencies() {
    print('State的didChangeDependencies来了');
    super.didChangeDependencies();
  }

  int _count = 0;

  @override
  Widget build(BuildContext context) {
    print('State的build来了');
    return Container(
      child: Column(
        children: [
          ElevatedButton(
            onPressed: () {
              _count++;
              setState(() {});
            },
            child: const Icon(Icons.add),
          ),
          Text('$_count'),
        ],
      ),
    );
  }

  @override
  void dispose() {
    print('State的dispose来了');
    super.dispose();
  }
}
  • 1.Widget构造方法
  • 2.WidgetCreateState
  • 3.State构造方法
  • 4.StateinitState
  • 5.didChangeDependencies方法(改变依赖关系);依赖(共享数据)的InheritedWidget发生变化之后,didChangeDependencies才会调用
  • 6.StateBuild ,当调用setState方法。会重新调用Build进行渲染,setState方法内部主要是利用_element(本质是就是context对象) 调用 markNeedsBuild
  • 7.当Widget销毁的时候调用Statedispose

2.3setState实现

  @protected
  void setState(VoidCallback fn) {
    assert(fn != null);
    assert(() {
      if (_debugLifecycleState == _StateLifecycle.defunct) {
        throw FlutterError.fromParts(<DiagnosticsNode>[
          ErrorSummary('setState() called after dispose(): $this'),
          ErrorDescription(
            'This error happens if you call setState() on a State object for a widget that '
            'no longer appears in the widget tree (e.g., whose parent widget no longer '
            'includes the widget in its build). This error can occur when code calls '
            'setState() from a timer or an animation callback.',
          ),
          ErrorHint(
            'The preferred solution is '
            'to cancel the timer or stop listening to the animation in the dispose() '
            'callback. Another solution is to check the "mounted" property of this '
            'object before calling setState() to ensure the object is still in the '
            'tree.',
          ),
          ErrorHint(
            'This error might indicate a memory leak if setState() is being called '
            'because another object is retaining a reference to this State object '
            'after it has been removed from the tree. To avoid memory leaks, '
            'consider breaking the reference to this object during dispose().',
          ),
        ]);
      }
      if (_debugLifecycleState == _StateLifecycle.created && !mounted) {
        throw FlutterError.fromParts(<DiagnosticsNode>[
          ErrorSummary('setState() called in constructor: $this'),
          ErrorHint(
            'This happens when you call setState() on a State object for a widget that '
            "hasn't been inserted into the widget tree yet. It is not necessary to call "
            'setState() in the constructor, since the state is already assumed to be dirty '
            'when it is initially created.',
          ),
        ]);
      }
      return true;
    }());
    final Object? result = fn() as dynamic;
    assert(() {
      if (result is Future) {
        throw FlutterError.fromParts(<DiagnosticsNode>[
          ErrorSummary('setState() callback argument returned a Future.'),
          ErrorDescription(
            'The setState() method on $this was called with a closure or method that '
            'returned a Future. Maybe it is marked as "async".',
          ),
          ErrorHint(
            'Instead of performing asynchronous work inside a call to setState(), first '
            'execute the work (without updating the widget state), and then synchronously '
            'update the state inside a call to setState().',
          ),
        ]);
      }
      // We ignore other types of return values so that you can do things like:
      //   setState(() => x = 3);
      return true;
    }());
    _element!.markNeedsBuild();
  }

setState方法其实核心就是最后一句_element!.markNeedsBuild();

  BuildContext get context {
    assert(() {
      if (_element == null) {
        throw FlutterError(
          'This widget has been unmounted, so the State no longer has a context (and should be considered defunct). \n'
          'Consider canceling any active work during "dispose" or using the "mounted" getter to determine if the State is still active.',
        );
      }
      return true;
    }());
    return _element!;
  }
  StatefulElement? _element;

由上面的源码可以得出_element其实就是Statebulid方法中的context

3.Flutter渲染原理

实际上,Flutter的UI绘制包含了三个元素,WidgetElementRenderObject。这三个元素分别组成了三棵树:Widget 树,Element 树和 RenderObject树,在Flutter渲染的流程中Flutter引擎渲染是针对Render树中的对象进行渲染

Flutter三棵树

系统启动时,runApp方法会被调用,Flutter会从最外层的Widget去遍历创建一颗Widget树;

  • 每一个Widget创建出来都会调用createElement方法创建一个Element对象
  • Element加入到Element树中,形成Element
  • Element通过mount方法调用createRenderObject创建RenderObject对象,并形成相应的RenderObject

StatelessElement 继承 ComponentElement主要调用build方法,并且将自己(Element)传递出去

StatefulElement 继承 ComponentElement,调用creatState方法,创建state,将Widget赋值给State对象,调用statebuild方法,并且将自己(Element)传出去

⚠️注意
只有继承自RenderObjectWidgetWidget才能创建RenderObject并加入RenderObject树,被渲染

3.1三棵树的作用

  • Widget树:配置信息,用来描述UI特征,比如尺寸多大,颜色是什么,位置在哪里
  • Element树:element是widget的实际实例,它同时持有了widget和renderObject的引用,用来决定是否进行UI更新。
  • RenderObject树:UI更新的执行者,保存了元素的大小,布局等信息。它才是真正调用渲染引擎去进行更新的对象
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 225,165评论 6 523
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 96,476评论 3 405
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 172,446评论 0 368
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 61,157评论 1 301
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 70,164评论 6 400
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 53,615评论 1 316
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 41,969评论 3 430
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 40,959评论 0 279
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 47,495评论 1 324
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 39,529评论 3 347
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 41,641评论 1 355
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 37,233评论 5 351
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,976评论 3 340
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 33,407评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 34,552评论 1 277
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 50,218评论 3 381
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 46,715评论 2 366