Dart的编译模式
Flutter 使用 Dart 作为编程语言,自然其编译模式也脱离不了 Dart 的干系。首先我们需要了解一下 Dart 所支持的编译模式。
Script:最普通的 JIT 模式,在 PC 命令行调用 Dart VM 执行 Dart 源代码文件即是这种模式;
Script Snapshot:JIT 模式,和上一个不同的是,这里载入的是已经 token 化的 Dart 源代码,提前执行了上一步的
lexer 步骤;
Application Snapshot:JIT 模式,这种模式来源于 Dart VM 直接载入源码后 dump 出数据。Dart VM
通过这种数据启动会更快。不过值得一提的是这种模式是区分架构的,在 x64 上生成的数据不可以给 arm 使用;
AOT:AOT模式,直接将 Dart 源码编译出 .S 文件,然后通过汇编器生成对应架构的代码。
Flutter的编译模式
Flutter 完全采用了 Dart, 由于 Android 和 iOS平台的生态差异,Flutter 也衍生出了非常丰富的编译模式。
Script:同 Dart Script 模式一致,虽然 Flutter 支持,但暂未看到使用,毕竟影响启动速度;
Script Snapshot:同 Dart Script Snapshot 一致,同样支持但未使用,Flutter
有大量的视图渲染逻辑,纯 JIT 模式影响执行速度;
Kernel Snapshot:Dart 的 bytecode模式,与 Application Snapshot 不同,bytecode
模式是不区分架构的。Kernel Snapshot 在 Flutter 项目内也叫 Core Snapshot。bytecode模式可以归类为 AOT 编译;
Core JIT:Dart 的一种二进制模式,将指令代码和 heap 数据打包成文件,然后在 VM 和 isolate
启动时载入,直接标记内存可执行,可以说这是一种 AOT 模式。Core JIT 也被叫做 AOTBlob;
AOT Assembly: 即 Dart 的 AOT 模式。直接生成汇编源代码文件,由各平台自行汇编
Android debug模式(Flutter使用Kernel Snapshot 模式编译)
运行flutter run(Debug模式) 安装app
解压apk包
安装apk成功后 查看data/data/pkg/app_flutter/flutter_assets目录 如下图
可以看出 产物中四个文件
isolate_snapshot_data:用于加速 isolate 启动,业务无关代码,固定,仅和 flutter engine 版本有关;
vm_snapshot_data: 用于加速 Dart VM 启动的产物,业务无关代码,仅和 flutter engine 版本有关;
platform.dill:和 Dart VM 相关的 kernel 代码,仅和 Dart 版本以及 engine 编译版本有关。固定,业务无关代码;
kernel_blob.bin:业务代码产物
Android release模式(Flutter使用Core JIT 模式编译)
通过flutter run --release (release模式)
解压apk包
安装apk成功后 查看data/data/pkg/app_flutter 如下图
进入flutter_asserts目录下 无内容
而app_flutter这一级 产物多出两个:isolate_snapshot_instr 和 vm_snapshot_instr,
它俩代表着 VM 和 isolate 启动后所承载的指令等数据。在载入后,直接将该块载入内存执行即可。
在release模式下 gen_snapshot是dart编译器
Dart代码 通过该编译器生成 vm/isolate_snapshot_data/instr
Custom engine 通过Ninja 构建系统来生成flutter.jar(包括了engine部分的代码(Flutter.jar中的libflutter.so),以及一套将Flutter嵌入Android的类和接口(FlutterMain,FlutterView,FlutterNativeView等))
总结:
事实上flutter下的iOS/Android工程本质上依然是一个标准的iOS/Android的工程,
iOS:flutter只是通过在BuildPhase中添加shell来生成和嵌入App.framework和Flutter.framework(iOS),
Android :通过gradle来添加flutter.jar和vm/isolate_snapshot_data/instr(Android)来将Flutter相关代码编译和嵌入原生App而已
以上有不正确之处,望指正
参考文章 https://blog.csdn.net/u010960265/article/details/81361711