打包器类型      
代码、资源打包 webpack vite(rollup/rolldown) parcel
代码打包 rollup/rolldown terser esbuild
编译器 babel(js) swc(wasm)  

对于前端打包工具有webpack、rollup等,既然有这些打包工具为什么还要在移动端搞一个metro,其中一个原因为ram bundle(按需加载的启动过程用到的模块),iOS采用indexed ram bundle读取一个文件效率更高,Android采用file ram bundle。除了支持 ram bundle 还能打出 hbc bundle(字节码 bundle)

那么接下来了解一下metro。

Metro生命周期

metro的bundling有三个阶段:

  1. 解析(Resolution): 解析所有模块并且构建成图,有点类似于Gradle在配置阶段会将所有相互依赖的任务构建成图。
  2. 转换(Transformation):转换阶段会将模块转换成目标平台能识别的格式,这一阶段执行了js编译,主流常用的js编译器为babel
  3. 序列化(Serialization):最后一个阶段序列化,会将所有转换之后的模块打包成一个或者多个bundle.

解析(Resolution)

在Gradle的配置阶段我们常看到assets、aidl、res、java的配置。

android{
    ...
    sourceSets {
        main {
            java.excludes = [
                    '**/build/**',
            ]
            srcDirs.forEach {
                assets.srcDirs += "$projectDir/$it/main/assets"
                aidl.srcDirs += "$projectDir/$it/main/aidl"
                res.srcDirs += "$projectDir/$it/main/res-frame-animation"
                res.srcDirs += "$projectDir/$it/main/res"
                java.srcDirs += "$projectDir/$it/main/java"
            }
        }
    }
    ...
}

metro与之对应项为assetExts、sourceExts。

转换(Transformation)

在ram bundle的启动优化中,通过getTransformOptions可以实现模块预加载,而其他的模块按需加载从而提高启动速度。

function getTransformOptions(
  entryPoints: $ReadOnlyArray<string>,
  options: {
    dev: boolean,
    hot: boolean,
    platform: ?string,
  },
  getDependenciesOf: (path: string) => Promise<Array<string>>,
): Promise<ExtraTransformOptions> {
  // ...
}

type ExtraTransformOptions = {
  preloadedModules?: {[path: string]: true} | false,
  ramGroups?: Array<string>,
  transform?: {
    inlineRequires?: {blockList: {[string]: true}} | boolean,
    nonInlinedRequires?: $ReadOnlyArray<string>,
  },
};

在preloadedModules中配置的模块为预加载模块,而其他的模块在ram bundle按需加载,这一块有点类似于Android multidex,Android5.0之前可以将部分类指明到主dex,其他被分配到辅dex。在Android App的构建流程中,编译完之后还会对字节码进行混淆,这块metro也有minifierPath(默认使用metro-minify-terser)、minifierConfig.在混淆这块除了terser,metro还提供了metro-minify-uglify。

序列化(Serialization)

在序列化的阶段模块需要有id以便于require导入,创建模块id的函数为createModuleIdFactory,而processModuleFilter决定了过滤掉哪些模块不进入bundle,所以通过createModuleIdFactory与processModuleFilter两个函数可以实现分包。

HMR Server

Hot React:Hot Module Replacement –> react-hot-loader –> react refresh

随着react-refresh、react-reconciler相继出现,react hot loader逐渐被替代,react refresh的实现与平台无关,React 、React Native等实现react-reconciler的自定义渲染器都能使用,而且react refresh能hot的颗粒度更小。在Web平台使用react refresh ,目前第三方开发者pmmmwh的项目react-refresh-webpack-plugin。移动平台则是React Native团队自己实现且内置到了metro打包器取名fast-refresh

参考资料

干货 | 减少50%空间,携程机票React Native Bundle 分析与优化

2020-06-02 React Native 打包工具Metro原理探究

ctripcorp/moles-packer

google/diff-match-patch

react-native-multibundler

React Native 拆包及实践「iOS&Android」

How should we set up apps for HMR now that Fast Refresh replaces react-hot-loader?

Fast Refresh原理剖析

webpack HMR

如果您觉得写得还不错或者对您有所启发,那就赶紧动动您的小指头,点击下面的红色按钮,狠狠地打赏一番吧。