《十六》微信小程序中的分包

微信小程序允许开发者将其划分为主包和多个分包,以优化启动时间和团队协作。主包包含默认启动页面和公共资源,分包按需加载。分包预下载和独立分包特性进一步提高用户体验。分包异步化则解决了跨分包组件和JS代码的依赖问题。配置分包预下载规则和使用组件占位符可确保流畅运行。

某些情况下,开发者需要将小程序划分成不同的子包,在构建时打包成不同的分包,用户在使用时按需进行加载。对小程序进行分包,可以优化小程序首次启动的下载时间,以及在多团队共同开发时可以更好的解耦协作。

在小程序启动时,默认会下载主包并启动主包内页面;当用户进入分包内某个页面时,客户端会把对应分包下载下来,下载完成后再进行展示。

目前小程序分包大小有以下限制:整个小程序所有分包大小不超过 20M,单个分包/主包大小不能超过 2M。

每个使用分包的小程序必需要包含有一个主包。主包放置默认启动页面、TabBar 页面以及一些所有分包都需用到公共资源/ JS 脚本;分包则是根据开发者的配置进行划分。

使用分包:

开发者通过在 app.json 中的 subpackages 字段声明项目的分包结构。

tabBar 页面必须在主包内。
不需要担心低版本的兼容问题。会由微信后台编译来处理旧版本客户端的兼容,后台会编译两份代码包,一份是分包后代码,另外一份是整包的兼容代码。 新客户端用分包,老客户端还是用的整包,完整包会把各个 subpackage 里面的路径放到 pages 中。

假设支持分包的小程序的目录结构如下:

在这里插入图片描述
那么要在 app.json 中进行如下配置:

{
  // 主包
  "pages":[
    "pages/index",
    "pages/logs"
  ],
  // 分包
  "subpackages": [
    {
      // 分包的根目录
      "root": "packageA",
      // 分包别名,分包预下载时可以使用
       "name": "pack1",
       // 分包页面路径,相对于分包根目录
      "pages": [
        "pages/cat",
        "pages/dog"
      ],
      // 分包是否是独立分包
      "independent": false
    }, 
    {
      "root": "packageB",
      "name": "pack2",
      "pages": [
        "pages/apple",
        "pages/banana"
      ],
      "independent": false
    }
  ]
}

独立分包:

独立分包可以独立于主包和其他分包运行。从独立分包中的页面进入小程序时,不需要下载主包。当用户进入普通分包或主包内页面时,主包才会被下载。

开发者通过 app.json 的 subpackages 字段中的 independent 字段声明独立分包。

独立分包属于分包的一种,普通分包的所有限制都对独立分包有效。
当用户从独立分包页面启动小程序时,主包不存在,App 也不存在,此时调用 getApp() 获取到的是 undefined。 当用户进入普通分包或主包内页面时,主包才会被下载,App 才会被注册,此时调用 getApp() 可以获取到真正的 App。
当从独立分包启动小程序时,主包中 App 的 onLaunch 和首次 onShow 会在从独立分包页面首次进入主包或其他普通分包页面时调用。

假设小程序目录结构如下:
在这里插入图片描述
那么在 app.json 中要进行如下配置:

{
  "pages": [
    "pages/index",
    "pages/logs"
  ],
  "subpackages": [
    {
      "root": "moduleA",
      "pages": [
        "pages/rabbit",
        "pages/squirrel"
      ]
    }, 
    {
      "root": "moduleB",
      "pages": [
        "pages/pear",
        "pages/pineapple"
      ],
      "independent": true
    }
  ]
}

分包预下载:

开发者可以通过配置,在进入小程序某个页面时,由框架自动预下载可能需要的分包,提升进入后续分包页面时的启动速度。

预下载分包行为在进入某个页面时触发,通过 app.json 中的 preloadRule 配置来控制。

同一个分包中的页面享有共同的预下载大小限额 2M,限额会在工具中打包时校验。
对于独立分包,也可以预下载主包。

{
  "pages": ["pages/index"],
  "subpackages": [
    {
      "root": "important",
      "pages": ["index"],
    },
    {
      "root": "sub1",
      "pages": ["index"],
    },
    {
      "root": "sub2",
      "pages": ["index"]
    },
    {
      "root": "sub3",
      "pages": ["index"]
    },
  ],
  "preloadRule": {
  	// key 是页面路径,value 是进入此页面的预下载配置
    "pages/index": {
      // 进入页面后预下载分包的 root 或 name。__APP__ 表示主包
      "packages": ["important"],
      // 在指定网络下预下载
      "network": "all"
    },
    "sub1/index": {
      "packages": ["sub2", "sub3"]
    },
  }
}

分包异步化:

默认情况下,除了非独立分包可以依赖主包外,分包之间不能互相使用自定义组件和 JS 代码。分包异步化使得部分跨分包的内容可以在等待下载后异步使用,从而在一定程度上解决了这个限制。

跨分包自定义组件引用:

一个分包使用其他分包的自定义组件时,可能由于其他分包还未下载或注入,导致其组件处于不可用的状态。

可以通过为其他分包的自定义组件设置占位组件,先渲染占位组件作为替代,在分包下载完成后再进行替换。

// subPackageA/pages/index.json
{
  "usingComponents": {
    "button": "../../commonPackage/components/button",
    "list": "../../subPackageB/components/full-list",
    "simple-list": "../components/simple-list",
  },
  "componentPlaceholder": {
  	// 要是 button 组件不可用,就先渲染 view 组件,等待 button 组件可用后再进行替换
    "button": "view",
    "list": "simple-list",
  }
}

跨分包 JS 代码引用:

一个分包中的代码引用其它分包的代码时,为了不让下载阻塞代码运行,需要异步获取引用的结果。

// subPackageA/index.js
// 使用回调函数风格的调用
require('../subPackageB/utils.js', utils => {
  console.log(utils.whoami) // Wechat MiniProgram
}, ({mod, errMsg}) => {
  console.error(`path: ${mod}, ${errMsg}`)
})

// 或者使用 Promise 风格的调用
require.async('../commonPackage/index.js').then(pkg => {
  pkg.getPackageName() // 'common'
}).catch(({mod, errMsg}) => {
  console.error(`path: ${mod}, ${errMsg}`)
})
<think> 我们正在讨论原生微信小程序分包功能。根据微信官方文档,分包是一种优化小程序加载速度的技术,允许开发者将小程序划分为多个子包,在需要时按需加载。 首先,我将简要介绍分包的概念和优势,然后详细说明使用方法(包括配置和代码示例),最后列举重要注意事项。 注意:所有技术术语和代码块需正确格式化,并在适当位置添加引用标识。 </think> ### 原生微信小程序分包指南 #### 一、分包介绍 分包是小程序优化的重要手段,允许开发者将小程序拆分成多个子包: 1. **主包**:包含启动页面、公共资源、核心逻辑等必需内容,大小上限**2MB** 2. **子包**:按功能划分的独立模块(如商品模块/用户中心),大小上限**20MB** ```text 小程序结构示例: ├── app.js (主包入口) ├── pages/ (主包页面) ├── subpackages/ │ ├── packageA/ (子包A) │ │ ├── pages/ │ │ └── components/ │ └── packageB/ (子包B) ``` **核心优势**: - **启动优化**:减少首屏加载时间[^1] - **突破体积限制**:总包大小可扩展至**20MB**(主包+所有子包) - **按需加载**:非核心功能延迟加载(如电商的商品详情页) - **团队协作**:不同团队可独立开发子包 #### 二、使用方法 ##### 1. 配置分包(app.json) ```json { "pages": [ "pages/index/index", "pages/logs/logs" ], "subpackages": [ { "root": "subpackages/shop", "name": "shop", "pages": [ "pages/goods/list", "pages/goods/detail" ] }, { "root": "subpackages/user", "pages": [ "pages/profile/index", "pages/settings/index" ] } ] } ``` - `root`:子包根目录(必填) - `name`:子包别名(选填) - `pages`:子包页面路径(相对于root) ##### 2. 分包预加载 在app.json中添加预加载配置: ```json "preloadRule": { "pages/index/index": { "network": "wifi", "packages": ["subpackages/shop"] } } ``` 当访问首页时自动预加载shop子包[^2] ##### 3. 分包跳转 使用`wx.navigateTo`跳转子包页面: ```javascript wx.navigateTo({ url: '/subpackages/shop/pages/goods/detail?id=123' }) ``` #### 三、注意事项 1. **目录规范** - 子包内不能包含主包资源 - 子包**不能嵌套**(不支持子包的子包) 2. **资源引用限制** - 主包可引用所有子包资源 - 子包**不能引用其他子包**的资源 - 公共组件建议放在主包 3. **体积管控** ```mermaid graph LR A[小程序总包] --> B(主包 ≤2MB) A --> C(子包A ≤20MB) A --> D(子包B ≤20MB) ``` 4. **特殊文件位置** - `app.js`/`app.json`/`app.wxss` 必须在主包 - `tabBar`页面必须在主包 5. **插件兼容** - 插件代码不计入分包大小 - 但插件自身资源需遵循相同引用规则 > 最佳实践:将使用频率低于20%的页面(如订单追踪/客服中心)放入子包,高频功能保留在主包[^3] --- ### 相关问题 1. 如何解决分包后公共组件复用问题? 2. 分包预加载策略有哪些优化技巧? 3. 小程序分包与独立分包有什么区别? 4. 分包后如何监控各子包的加载性能? [^1]: 微信官方文档:分包加载原理 [^2]: 微信开发者工具预加载配置说明 [^3]: 小程序性能优化白皮书第3章
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值