# 使用分包

作为一个对 performance 极度重视的框架,分包作为提升小程序体验的重要能力,框架对各种类型的分包能力进行了完善支持。

分包是小程序平台提供的原生能力,mpx是对该能力做了部分加强,目前各大主流小程序平台都已支持分包,且框架在可能的情况下进行了抹平。

使用分包一定要记得阅读下面的分包注意事项

# 原生语法注册分包

Mpx 支持小程序原生语法注册分包,并且在框架层面对不同平台的的差异进行了抹平,我们以微信小程序原生语法注册分包为例。

{
  "pages": [
    "pages/index/index"
  ],
  "subPackages": [
    {
      "root": "test",
      "pages": [
        "pages/other/other",
        "pages/other/other2"
      ]
    }
  ]
}

# packages 语法注册分包

Mpx 提供 packages 语法来对小程序的主包页面和分包划分能力进行增强,使用 packages 语法可以灵活的对业务进行拆分,允许以 npm 包的形式进行 主包页面和分包注册,且分包名和页面路径可自定义,十分有利于大型多团队开发的项目维护。

# 使用方法

Mpx 拓展了 app.json 的语法,新增了 packages 域,用来声明依赖的 packages,packages 可嵌套依赖。

# 注册主包页面

首先我们介绍下 packages 注册主包页面的用法,在 packages 中直接配置资源路径,Mpx 会去读取该资源中 json 区块中的 pages 属性,合并到主包页面配置中。

// @file src/app.mpx
<script type="application/json">
  {
    "pages": [
      "./pages/index/index"
    ],
    "packages": [
      "{npmPackage || relativePathToPackage}/index"
    ]
  }
</script>

// @file src/packages/index.mpx
// 注意确保页面路径的唯一性
<script type="application/json">
  {
    "pages": [
      "./pages/other/other",
      "./pages/other/other2"
    ]
  }
</script>

打包结果:dist/app.json

{
  "pages": [
    "pages/index/index",
    "pages/other/other",
    "pages/other/other2"
  ]
}

# 注册分包

Mpx 会将 packages 域下的路径带 root 为 key 的 query 解析注册为分包,使用 packages 语法注册分包,只需要在 packages 中配置资源路径添加 root=xxx,root的值即为分包名。

// @file src/app.mpx
<script type="application/json">
  {
    "pages": [
      "./pages/index/index"
    ],
    "packages": [
      "{npmPackage || relativePathToPackage}/index?root=test"
    ]
  }
</script>

// @file src/packages/index.mpx (子包的入口文件)
<script type="application/json">
  {
    "pages": [
      "./pages/other/other",
      "./pages/other/other2"
    ]
  }
</script>

打包结果:dist/app.json

{
  "pages": [
    "pages/index/index"
  ],
  "subPackages": [
    {
      "root": "test",
      "pages": [
        "pages/other/other",
        "pages/other/other2"
      ]
    }
  ]
}

由上可见,经过我们的编译过程,packages 中注册的页面可按照原始的路径被合并入主包页面或注册为分包, 这样开发者可以不用考虑自己在被依赖时页面路径是怎么样的,也可以直接将调试用的app.mpx作为依赖入口直接暴露出去, 对于主app的开发者来说也不需要了解依赖内部的细节,只需要在packages中声明自己所需的依赖即可。

# 注意事项

  • 依赖的开发者在自己的入口 app.mpx 中注册页面时对于本地页面一定要使用相对路径进行注册,否则在主app中进行编译时会找不到对应的页面
  • 不管是用 json 还是 mpx 格式定义 package 入口,编译时永远只会解析 json 且只会关注 json 中的 pages 和 packages 域,其余所有东西在主app编译时都会被忽略
  • 由于我们是将 packages 中注册的页面按照原始的路径合并到主 app 当中,有可能会出现路径名冲突。
    这种情况下编译会报出响应错误提示用户解决冲突,为了避免这种情况的发生,依赖的提供者最好将自己内部的页面放置在能够描述依赖特性的子文件夹下。

例如一个包叫login,建议包内页面文件目录为:

project
│   app.mpx  
└───pages
    └───login
        │   page1.mpx
        │   page2.mpx
        │   ...

# 独立分包

Mpx目前已支持独立分包构建,使用 packages 语法声明分包时只需要在后面添加 independent=true query 即可,同时也支持原生语法声明。 如下方示例声明 packageA 分包为独立分包

示例:

// src/app.mpx 文件中 json 块

// Mpx packages 方式
{
  "packages": [
    "packageA/app.mpx?root=packageA&independent=true"
  ]
}
// 微信原生方式
{
  "subpackages": [
    {
      "root": "packageA",
      "pages": [
        "pages/index"
      ],
      "independent": true
    },
  ]
}

需要注意的是,由于独立分包可以独立于主包和其他分包运行,从独立分包页面进入小程序时,主包中的相应初始化逻辑并不会执行,如果独立分包中多个页面需要某种通用初始化逻辑时就无法优雅的实现, Mpx框架针对独立分包场景提供了独立分包初始化逻辑执行能力。

对于使用 packages 方式声明的独立分包,默认将 .mpx 文件自身的 script 块作为初始化逻辑执行。

<!--src/packagesA/app.mpx,packageA 独立分包入口文件-->
<script>
import mpx from '@mpxjs/core'
import apiProxy from '@mpxjs/api-proxy'

mpx.use(apiProxy, { usePromise: true }) 
if (isIndependent) {
    // do some in independent package
} else {
    // do some not independent package
}
</script>

<script type="application/json">
{
  "pages": [
    "./pages/index"
  ]
}
</script>

上方代码中 独立分包 packageA 的入口文件 app.mpx 中的 script block 代码会默认在独立分包初始化时执行,Mpx 同时提供了全局变量 isIndependent 标识当前代码执行环境是否为独立分包来进行特定逻辑区分

如果你不想走这个默认的初始化逻辑执行规则,想自定义一个 js 文件存储当前独立分包的初始化逻辑,我们支持 independent 配置项直接配置为初始化逻辑文件地址

// src/app.mpx 文件中 json 块
// Mpx packages 方式
{
  "packages": [
    "packageA/app.mpx?root=packageA&independent=./common" // 路径上下文为 packageA 文件夹
  ]
}
// 微信原生方式
{
  "subpackages": [
    {
      "root": "packageA",
      "pages": [
        "pages/index"
      ],
      "independent": "./common" // 路径上下文为 packageA 文件夹
    },
  ]
}
// src/pacakgeA/common.js
import mpx from '@mpxjs/core'
import apiProxy from '@mpxjs/api-proxy'

mpx.use(apiProxy, { usePromise: true })
if (isIndependent) {
    // do some in independent package
} else {
    // do some not independent package
}

注意上方配置 independent 为初始化逻辑文件地址时,路径相对地址上下文为 packageA

# 分包预下载

分包预下载是在 json中 新增一个 preloadRule 字段,mpx 打包时候会原封不动把这个部分放到 app.json 中,所以只需要按照 微信小程序官方文档 - 分包预下载 (opens new window) 或者 支付宝小程序官方文档 - 分包预下载 (opens new window) 配置即可。

示例:

// @file src/app.mpx
<script type="application/json">
  {
    "pages": [
      "./pages/index/index"
    ],
    "packages": [
      "{npmPackage || relativePathToPackage}/index?root=xxx"
    ],
    "preloadRule": {
      "pages/index": {
        "network": "all",
        "packages": ["important"]
      },
      "sub1/index": {
        "packages": ["hello", "sub3"]
      }
    }
  }
</script>

// @file src/packages/index.mpx (子包的入口文件)
<script type="application/json">
  {
    "pages": [
      "./pages/other/other",
      "./pages/other/other2"
    ]
  }
</script>

打包结果:dist/app.json

{
  "pages": [
    "pages/index/index"
  ],
  "subPackages": [
    {
      "root": "xxx",
      "pages": [
        "pages/other/other",
        "pages/other/other2"
      ]
    }
  ],
  "preloadRule": {
    "pages/index": {
      "network": "all",
      "packages": ["important"]
    },
    "sub1/index": {
      "packages": ["hello", "sub3"]
    }
  }  
}

# 分包注意事项

当我们使用分包加载时,依赖包内的跳转路径需注意,比如要跳转到other2页面

  • 不用分包时会是:/pages/other/other2
  • 使用分包后应为:/test/pages/other/other2

即前面会多?root={rootKey}的rootKey这一层

为了解决这个问题,有三种方案:

  • import的时候在最后加'?resolve', 例如: import testPagePath from '../pages/testPage.mpx?resolve' , 编译时就会把它处理成正确的完整的绝对路径。

  • 使用相对路径跳转。

  • 定死使用的分包路径名,直接写/{rootKey}/pages/xxx (极度不推荐,尤其在分包可能被多方引用的情况时)

这里我们建议使用第一种方式。