MXFlutter动态化方案分析

前言

如今Flutter框架越来越趋于成熟,对于跨端渲染一致性问题有极致要求的项目可以更方便地迁移到Flutter框架上,而对于那些还在犹豫的项目组最大的阻碍可能就是动态化的解决方案上了。MXFlutter(https://juejin.im/post/5d11a4f06fb9a07ec63b21ea
)为我们提供了一个很好的思路,最近由于项目上的需求,我对该方案也做了一些调研,下面我们就从源码层面粗略地分析一下MXFlutter动态化方案的原理,希望能抛砖引玉引出更多优秀的点子。

Widget->Element->RenderObject

我们先来分析一下一个最普通的用dart编写的Flutter Widget是怎么工作的:

import 'package:flutter/material.dart';

void main() => runApp(new MyApp());

class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return new MaterialApp(
      title: 'Welcome to Flutter',
      home: new Scaffold(
        appBar: new AppBar(
          title: new Text('Welcome to Flutter'),
        ),
        body: new Center(
          child: new Text('Hello World'),
        ),
      ),
    );
  }
}

这是一个最简单的Flutter Hello World程序,我们先从main函数看起,里面调用了WidgetBinding的runApp方法,WidgetBinding是Widget框架和Flutter引擎的胶水层(The glue between the widgets layer and the Flutter engine.)。在runApp方法中调用attachRootWidget将我们定义的MyApp作为根布局添加到屏幕上:

void runApp(Widget app) {
  WidgetsFlutterBinding.ensureInitialized()
    ..attachRootWidget(app)
    ..scheduleWarmUpFrame();
}

紧接着在attachRootWidget中又会通过RenderObjectToWidgetAdapter来最终把Element与RenderObject结合起来。通过官方注释我们也可以了解到RenderObjectToWidgetAdapter是Element与RenderObject之间的一座桥梁:

RenderObjectToWidgetAdapter: A bridge from a RenderObject to an Element tree.

  void attachRootWidget(Widget rootWidget) {
    _renderViewElement = RenderObjectToWidgetAdapter<RenderBox>(
      container: renderView,
      debugShortDescription: '[root]',
      child: rootWidget,
    ).attachToRenderTree(buildOwner, renderViewElement);
}

我们接着看attachToRenderTree方法:

RenderObjectToWidgetElement<T> attachToRenderTree(BuildOwner owner, [ RenderObjectToWidgetElement<T> element ]) {
    if (element == null) {
      owner.lockState(() {
        element = createElement();
        assert(element != null);
        element.assignOwner(owner);
      });
      owner.buildScope(element, () {
        element.mount(null, null);
      });
    } else {
      element._newWidget = this;
      element.markNeedsBuild();
    }
    return element;
}

在这里就是Element被创建的地方,而这个createElement方法则是每个具体的Widget类自己实现的,以Flutter的StatelessWidget为例:

/// Creates a [StatelessElement] to manage this widget's location in the tree.
///
/// It is uncommon for subclasses to override this method.
@override
StatelessElement createElement() => StatelessElement(this);

我肯可以看到在createElement会把Widget的引用传给Element。在Element的preformBuild会调用到Element的build方法,而Element的build实际是调用了Widget的build:

/// Calls the [StatelessWidget.build] method of the [StatelessWidget] object
  /// (for stateless widgets) or the [State.build] method of the [State] object
  /// (for stateful widgets) and then updates the widget tree.
  ///
  /// Called automatically during [mount] to generate the first build, and by
  /// [rebuild] when the element needs updating.
  @override
  void performRebuild() {
    ···
    Widget built;
    try {
      built = build();
      debugWidgetBuilderValue(widget, built);
    } catch (e, stack) {
    ···
    } finally {
    ···
    }
    ···
  }
class StatelessElement extends ComponentElement {
    @override
    Widget build() => widget.build(this);
}
class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    ···
  }
}

而Widget的build方法,则是我们在开头看到的需要业务侧重写的具体的页面布局。这里有一个细节,Widget的build接收的参数是BuildContext,而我们传入的是Element,查看源码就可以发现,Element其实是继承了BuildContext。至此Widget->Element->RenderObject在代码层面的调用关系大概就理清楚了。

MXFlutter原理介绍

核心思路是把 Flutter 的渲染逻辑中的三棵树中的第一棵,放到 JavaScript 中生成。用 JavaScript 完整实现了 Flutter 控件层封装,可以使用 JavaScript,用极其类似 Dart 的开发方式,开发Flutter应用,利用JavaScript版的轻量级Flutter Runtime,生成UI描述,传递给Dart层的UI引擎,UI引擎把UI描述生产真正的 Flutter 控件。——MXFlutter

以MXFlutter Android端Demo为例,该项目集成了J2V8(https://eclipsesource.com/blogs/tutorials/getting-started-with-j2v8/
),这个库的作用是在没有WebView环境下,也能方便地进行JS与Java之间的相互通信。

在MXFlutter里通过用JS维护了一颗假的Widget树,然后通过JS-Java-Dart映射到Dart侧的Widget树。我们可以看到在Dart侧的MXJSWidget只是一个容器,内部封装了JS侧的MXJSWidget

 @override
  Widget build(BuildContext context) {
    MXJSLog.log("MXJSWidget:build: ${widget.widgetData} ");
    if (widget.widgetData == null) {
      return _buildErrorWidget();
    }
    var w = _jsonBuildOwner.build(widget.widgetData, context);

    //告诉JS层,使用当前JSWidget 序列号的数据构建,callbackID,widgetID  与之对应
    _jsonBuildOwner.callJSOnBuildEnd();

    return w;
  }

在启动MXFlutter的Demo时,会先启动一个真实的Dart Widget——main.dart,在点击JSFlutter UI Demo后,会调用MXJSFlutter的navigatorPushWithPageName方法,通过内部一系列的方法调用最终返回一个MXJSWidget

 //flutter层 主动push页面
 MXJSWidget navigatorPushWithPageName(String widgetName, {ThemeData themeData, MediaQueryData mediaQueryData, IconThemeData iconThemeData}) {

    MXJSWidget jsWidget = currentApp?.navigatorPushWithPageName(widgetName,
        themeData: themeData, mediaQueryData: mediaQueryData, iconThemeData: iconThemeData);

    return jsWidget;
 }
//push js页面
//先创建一个空的MXJSWidget,调用JS,等待JS层widgetData来刷新页面
  MXJSWidget navigatorPushWithPageName(String widgetName,
      {ThemeData themeData, MediaQueryData mediaQueryData, IconThemeData iconThemeData}) {
    ···
    callJSNavigatorPushWithPageName(widgetName,
        themeData: themeData, mediaQueryData: mediaQueryData, iconThemeData: iconThemeData);

    firstBuildWidget = jsWidget;

    return jsWidget;
  }
 //flutter层 主动push页面
  callJSNavigatorPushWithPageName(String widgetName,
      {ThemeData themeData, MediaQueryData mediaQueryData, IconThemeData iconThemeData}) async {
    MethodCall jsMethodCall =
        MethodCall("flutterCallNavigatorPushWithPageName", {
      "pageName": name,
      "themeData": MXUtil.cThemeDataToJson(themeData),
      "mediaQueryData": MXUtil.cMediaQueryDataToJson(mediaQueryData),
      "iconThemeData": MXUtil.cIconThemeDataToJson(iconThemeData),
    });

    callJS(jsMethodCall);
  }
void callJS(MethodCall jsMethodCall) {
    MXJSLog.log("callJSWidget:$jsMethodCall");

    var jsArgs = {
      "method": jsMethodCall.method,
      "arguments": jsMethodCall.arguments,
    };

    _jsFlutterAppChannel.invokeMethod("callJS", jsArgs);
}

等待JS侧执行完毕后会返回一个widgetData的json结构,dart侧则根据这个数据结构构建出相应的widget

 //js->flutter 显示js页面
  Future<dynamic> reloadApp(args) async {
    String routeName = args["routeName"];

    if (routeName == "MXJSWidget") {
      String widgetDataStr = args["widgetData"];
      var widgetData = json.decode(widgetDataStr);

      try {
        var w = currentApp.createJSWidget(widgetData);
        currentApp.runJSApp(w);
      } catch (e) {
        MXJSLog.log("reloadApp error:$e");
        throw (e);
      }
    } else {
      //runApp(MyApp());
    }
  }

在createJSWidget方法中一路调用最终会走到MXJsonObjToDartObject的jsonObjToDartObject方法:

  dynamic jsonObjToDartObject(MXJsonBuildOwner buildOwner, dynamic jsonObj, {BuildContext context}) {
    String className;
     try {
      ///map
      if (jsonObj is Map) {
        className = getJsonObjClassName(jsonObj);

        ///如果Map里找到了Class字段,则转换成对应Dart对象
        if (className != null) {
          return jsonMapObjToDartObject(buildOwner, jsonObj, context:context);
        } else {
          ///如果Map里没找到Class字段,则转换成对应Dart里的Map对象,并对齐子元素,递归转换
          return jsonMapObjToDartMapRecursive(buildOwner, jsonObj, context:context);
        }
      } else if (jsonObj is List) {
        return jsonListObjToDartListRecursive(buildOwner, jsonObj, context:context);
      } else {
        return jsonObj;
      }
    } catch (e) {
      MXJSLog.error(
          "MXJsonObjToDartObject:jsonObjToDartObject error:$e ;decode:class $className, jsonObj:$jsonObj ");

          //打印日志重新抛出
          throw e;
    }
  }

后记

这套方案通过JSCore代替DartVM,使用JS来写Widget,从而实现了动态化。但至于MXFlutter所宣传单“高性能”,我觉得还是有待商榷,比起原生的Flutter,JS与Native、Dart间的通信增加了不小的额外开销。同时由于J2V8库的引入,对于包体积也会有不小的增加,但这的确是一条可行的动态化思路。后续如果能持续维护的话,增加对于Vue、RN的DSL转换也是能有很大的现实意义。

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

推荐阅读更多精彩内容