Flutter局部刷新原理

  • 概述

    在Flutter中,我们知道,刷新界面要调用setState方法,在一个界面中,通常只需要刷新某个组件或者某一部分组件,这种情况下调用父级State的setState方法会造成不必要的资源浪费。 在这种需求下,我们需要找到一个方式可以进行局部刷新。

  • 做法和原理

    其实局部刷新很简单,我们只需要把需要刷新的组件聚到一个StatefulWidget中,通过一个State来管理,然后刷新的时候调用这个State的setState方法即可完成针对这部分组件的刷新,父级和兄弟级的StatefulWidget都不会被引起rebuild。

    原理也很好理解,就是调用setState的流程,调用setState方法:

    @protected
    void setState(VoidCallback fn) {
      ...
      _element!.markNeedsBuild();
    }
    

    Element的markNeedsBuild方法会调用BuildOwner的scheduleBuildFor方法:

    void markNeedsBuild() {
      ...
      if (dirty)
        return;
      _dirty = true;
      owner!.scheduleBuildFor(this);
    }
    

    scheduleBuildFor方法会把当前element放入BuildOwner的_dirtyElements中:

    void scheduleBuildFor(Element element) {
      ...
      _dirtyElements.add(element);
      element._inDirtyList = true;
      ...
    }
    

    当下一个Frame到来时框架会调用WidgetsBinding的drawFrame方法:

    @override
    void drawFrame() {
      ...
      try {
        if (renderViewElement != null)
          buildOwner!.buildScope(renderViewElement!);
        //这里面是布局、合成层信息、绘制等流程
        super.drawFrame();
        buildOwner!.finalizeTree();
      } finally {
          ...
      }
      ...
    }
    

    这里会调用BuildOwner的buildScope方法:

    @pragma('vm:notify-debugger-on-exception')
    void buildScope(Element context, [ VoidCallback? callback ]) {
      ...
      try {
        ...
        _dirtyElements.sort(Element._sort);
          ...
        int dirtyCount = _dirtyElements.length;
        int index = 0;
        while (index < dirtyCount) {
          ...
          try {
            //重新构建
            _dirtyElements[index].rebuild();
          } catch (e, stack) {
            ...
          }
          index += 1;
          ...
        }
          ...
      } finally {
        for (final Element element in _dirtyElements) {
          assert(element._inDirtyList);
          element._inDirtyList = false;
        }
        //清空_dirtyElements
        _dirtyElements.clear();
        ...
      }
      ...
    }
    

    在buildScope方法中会循环 _dirtyElements,依次调用里面的element的rebuild方法进行构建,rebuild方法中又会调用performRebuild方法:

    @pragma('vm:prefer-inline')
    void rebuild() {
      ...
      performRebuild();
      ...
    }
    

    performRebuild方法是在StatefulElement和StatelessElement的共同父类ComponentElement中实现的,在这个方法中会调用build方法创建Widget:

    //StatefulElement中实现的build方法,可见会通过state的build方法生成
    @override
    Widget build() => state.build(this);
    //StatelessElement中实现的build方法
    @override
    Widget build() => widget.build(this);
    

    所以局部刷新原理的核心就是把需要刷新的区域收到一个State中,然后调用这个State的setState方法就会使当前的这个State的element变为dirty,把它放入需要重新构建的element集合中,在帧回调后会循环这个集合调用它的rebuild方法进行重新构建,因为我们更上一级的State并没有执行它的setState方法所以不会添加在需要重新构建的element集合中。

  • 关于get框架的应用

    get框架的局部刷新也是通过上面的原理完成的,下面我们来看看他是怎么封装的。

    首先它使用一个叫做GetxController的东西来提供统一刷新的api接口:

    abstract class GetxController extends DisposableInterface
        with ListenableMixin, ListNotifierMixin {
      void update([List<Object>? ids, bool condition = true]) {
        if (!condition) {
          return;
        }
        //全部刷新
        if (ids == null) {
          refresh();
        } else {
          //局部刷新
          for (final id in ids) {
            refreshGroup(id);
          }
        }
      }
    }
    

    在页面打开的时候会创建这个controller,然后通过调用这个controller的update方法执行局部构建,可以看到,局部构建需要一个id,这个id是什么时候绑定的呢?

    使用get框架的局部刷新需要把要刷新的组件们用一个GetBuilder包装起来,那这个GetBuilder构造时就可以传入一个id值,GetBuilder是一个StatefulWidget,他的State中的initState方法里调用了一个_subscribeToController方法:

    void _subscribeToController() {
      _remove?.call();
      _remove = (widget.id == null)
          //全部刷新的回调添加
          ? controller?.addListener(
              _filter != null ? _filterUpdate : getUpdate,
            )
          //局部刷新的回调添加
          : controller?.addListenerId(
              widget.id,
              _filter != null ? _filterUpdate : getUpdate,
            );
    }
    

    addListenerId方法中:

    Disposer addListenerId(Object? key, GetStateUpdate listener) {
      _updatersGroupIds![key] ??= <GetStateUpdate>[];
      _updatersGroupIds![key]!.add(listener);
      return () => _updatersGroupIds![key]!.remove(listener);
    }
    

    可以看到,这里根据id添加了一个回调函数,这里用的数组存放,可见可以通过指定同一个id的方式来实现几个区域联动刷新。

    回到上面的refreshGroup方法,内部会调用_notifyIdUpdate方法:

    void _notifyIdUpdate(Object id) {
      if (_updatersGroupIds!.containsKey(id)) {
        final listGroup = _updatersGroupIds![id]!;
        for (var item in listGroup) {
          item();
        }
      }
    }
    

    可见,在这里根据id查找并执行了所有相关的函数回调。

    那么函数回调是什么呢?_subscribeToController方法中,addListenerId方法添加的函数回调如果默认的话是getUpdate,它指向一个函数,这个函数在GetBuilderState依赖的mixin—GetStateUpdaterMixin中定义:

    void getUpdate() {
      if (mounted) setState(() {});
    }
    

    可以看到,正是在这里调用了setState来触发重新构建的,因为是在GetBuilderState中调用的setState方法,所以在GetBuilder之上的其他State是不会触发回调的,这和上面我们分析的原理是一样的。

  • 总结

    局部构建的原理就是用子State来拦截构建的范围,不把所有的组件树都放在一个大的State里面构建,通过调用子State的setState方法来实现针对子Widget树的重新构建,这样就实现了局部刷新。

    据此,我们当然可以不用get框架的局部刷新,完全可以自定义,我试着写了一下,有几个需要注意的点:

    1. setState一定要在需要局部刷新的State中调用;

    2. 调用setState的逻辑要通过一个函数暴露出来;

    3. 因为我们要保证随时可以刷新,所以我们需要一个随时获取且不会改变的对象来保存这个回调,相当于get的controller;

    4. 因为我们可能会有很多个局部需要刷新,它们必须独立且可以区分,所以我们需要保存回调函数的集合是一个可已按照key-value的形式来存放的集合,get中使用了String-List的形式,这样可以刷新好几块区域,我用的是一个String-dynamic的Map来存放,这个过程中发现了一个需要注意的点:

      Map的putIfAbsent方法的第二个参数规定是:

      V putIfAbsent(K key, V ifAbsent());
      

      如果使用这个方法设置回调函数,则需要在一个函数中返回这个回调函数才行。

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

推荐阅读更多精彩内容