微前端解决方案
任意 js 框架均可使用。子应用接入像使用接入一个 iframe 系统一样简单,但实际不是 iframe。
几乎包含所有构建微前端系统时所需要的基本能力,如 样式隔离、js 沙箱、预加载等。
已在内外经受过足够大量的线上系统的考验及打磨,健壮性值得信赖。
Techniques, strategies and recipes for building a modern web app with multiple teams that can ship features independently. -- Micro Frontends
微前端是一种多个团队通过独立发布功能的方式来共同构建现代化 web 应用的技术手段及方法策略。
微前端架构具备以下几个核心价值:
- 技术栈无关
主框架不限制接入应用的技术栈,子应用具备完全自主权 - 独立开发、独立部署
子应用仓库独立,前后端可独立开发,部署完成后主框架自动完成同步更新 - 增量升级
在面对各种复杂场景时,我们通常很难对一个已经存在的系统做全量的技术栈升级或重构,而微前端是一种非常好的实施渐进式重构的手段和策略 - 独立运行时
每个子应用之间状态隔离,运行时状态不共享
微前端架构旨在解决单体应用在一个相对长的时间跨度下,由于参与的人员、团队的增多、变迁,从一个普通应用演变成一个巨石应用(Frontend Monolith)后,随之而来的应用不可维护的问题。这类问题在企业级 Web 应用中尤其常见
更多关于微前端的相关介绍,推荐大家可以去看这几篇文章:
- 基于 single-spa 封装,提供了更加开箱即用的 API
- 技术栈无关,任意技术栈的应用均可 使用/接入,不论是 React/Vue/Angular/JQuery 还是其他等框架。
- HTML Entry 接入方式,让你接入子应用像使用 iframe 一样简单。
- 样式隔离,确保子应用之间样式互相不干扰。
- JS 沙箱,确保子应用之间 全局变量/事件 不冲突。
- 资源预加载,在浏览器空闲时间预加载未打开的子应用资源,加速子应用打开速度。
npm i legions-micro-service --save
import { registerMicroApps, start } from 'legions-micro-service';
// 注册子应用
registerMicroApps([
{
name: 'reactApp',
entry: '//localhost:7100',
container: '#subapp-viewport',
loader: ({ appContent }) => ReactDOM.render(<App appContent={appContent}>, document.getElementById('container')),
activeRule: '/demo3#/react15',
},
]);
start();
适用于 route-based 场景。
通过将微应用关联到一些 url 规则的方式,实现当浏览器 url 发生变化时,自动加载相应的微应用的功能。
-
参数
- apps -
Array<RegistrableApp>
- 必选,微应用的一些注册信息 - lifeCycles -
LifeCycles
- 可选,全局的微应用生命周期钩子
- apps -
-
类型
-
RegistrableApp
-
name -
string
- 必选,微应用的名称,微应用之间必须确保唯一。 -
entry -
string | { scripts?: string[]; styles?: string[]; html?: string }
- 必选,微应用的入口。- 配置为字符串时,表示微应用的访问地址,例如
https://legions.umijs.org/guide/
。 - 配置为对象时,
html
的值是微应用的 html 内容字符串,而不是微应用的访问地址。微应用的publicPath
将会被设置为/
。
- 配置为字符串时,表示微应用的访问地址,例如
-
container -
string | HTMLElement
- 必选,微应用的容器节点的选择器或者 Element 实例。如container: '#root'
或container: document.querySelector('#root')
。 -
activeRule -
string | (location: Location) => boolean | Array<string | (location: Location) => boolean>
- 必选,微应用的激活规则。- 支持直接配置字符串或字符串数组,如
activeRule: '/app1'
或activeRule: ['/app1', '/app2']
,当配置为字符串时会直接跟 url 中的路径部分做前缀匹配,匹配成功表明当前应用会被激活。 - 支持配置一个 active function 函数或一组 active function。函数会传入当前 location 作为参数,函数返回 true 时表明当前微应用会被激活。如
location => location.pathname.startsWith('/app1')
。
规则示例:
'/app1'
'/users/:userId/profile'
- ✅ https://app.com/users/123/profile
- ✅ https://app.com/users/123/profile/sub-profile/
- 🚫 https://app.com/users//profile/sub-profile/
- 🚫 https://app.com/users/profile/sub-profile/
'/pathname/#/hash'
- ✅ https://app.com/pathname/#/hash
- ✅ https://app.com/pathname/#/hash/route/nested
- 🚫 https://app.com/pathname#/hash/route/nested
- 🚫 https://app.com/pathname#/another-hash
['/pathname/#/hash', '/app1']
- ✅ https://app.com/pathname/#/hash/route/nested
- ✅ https://app.com/app1/anything/everything
- 🚫 https://app.com/pathname/app1
- 🚫 https://app.com/app2
浏览器 url 发生变化会调用 activeRule 里的规则,
activeRule
任意一个返回true
时表明该微应用需要被激活。 - 支持直接配置字符串或字符串数组,如
-
loader -
(loading: boolean) => void
- 可选,loading 状态发生变化时会调用的方法。 -
props -
object
- 可选,主应用需要传递给微应用的数据。
-
-
LifeCycles
type Lifecycle = (app: RegistrableApp) => Promise<any>;
- beforeLoad -
Lifecycle | Array<Lifecycle>
- 可选 - beforeMount -
Lifecycle | Array<Lifecycle>
- 可选 - afterMount -
Lifecycle | Array<Lifecycle>
- 可选 - beforeUnmount -
Lifecycle | Array<Lifecycle>
- 可选 - afterUnmount -
Lifecycle | Array<Lifecycle>
- 可选
- beforeLoad -
-
-
用法
注册微应用的基础配置信息。当浏览器 url 发生变化时,会自动检查每一个微应用注册的
activeRule
规则,符合规则的应用将会被自动激活。 -
示例
import { registerMicroApps } from 'legions-micro-service'; registerMicroApps( [ { name: 'app1', entry: '//localhost:8080', container: '#container', activeRule: '/react', props: { name: 'kuitos', } } ], { beforeLoad: app => console.log('before load', app.name), beforeMount: [ app => console.log('before mount', app.name), ], }, );
-
参数
- opts -
Options
可选
- opts -
-
类型
-
Options
-
prefetch -
boolean | 'all' | string[] | (( apps: RegistrableApp[] ) => { criticalAppNames: string[]; minorAppsName: string[] })
- 可选,是否开启预加载,默认为true
。配置为
true
则会在第一个微应用 mount 完成后开始预加载其他微应用的静态资源配置为
'all'
则主应用start
后即开始预加载所有微应用静态资源配置为
string[]
则会在第一个微应用 mounted 后开始加载数组内的微应用资源配置为
function
则可完全自定义应用的资源加载时机 (首屏应用及次屏应用) -
sandbox -
boolean
|{ strictStyleIsolation?: boolean, experimentalStyleIsolation?: boolean }
- 可选,是否开启沙箱,默认为true
。默认情况下沙箱可以确保单实例场景子应用之间的样式隔离,但是无法确保主应用跟子应用、或者多实例场景的子应用样式隔离。当配置为
基于 ShadowDOM 的严格样式隔离并不是一个可以无脑使用的方案,大部分情况下都需要接入应用做一些适配后才能正常在 ShadowDOM 中运行起来(比如 react 场景下需要解决这些 问题,使用者需要清楚开启了{ strictStyleIsolation: true }
时表示开启严格的样式隔离模式。这种模式下 legions 会为每个微应用的容器包裹上一个 shadow dom 节点,从而确保微应用的样式不会对全局造成影响。strictStyleIsolation
意味着什么。后续 legions 会提供更多官方实践文档帮助用户能快速的将应用改造成可以运行在 ShadowDOM 环境的微应用。除此以外,qiankun 还提供了一个实验性的样式隔离特性,当 experimentalStyleIsolation 被设置为 true 时,qiankun 会改写子应用所添加的样式为所有样式规则增加一个特殊的选择器规则来限定其影响范围,因此改写后的代码会表达类似为如下结构:
// 假设应用名是 react16 .app-main { font-size: 14px; } div[data-legions-react16] .app-main { font-size: 14px; }
注意: @keyframes, @font-face, @import, @page 将不被支持 (i.e. 不会被改写)
-
singular -
boolean | ((app: RegistrableApp<any>) => Promise<boolean>);
- 可选,是否为单实例场景,单实例指的是同一时间只会渲染一个微应用。默认为true
。 -
fetch -
Function
- 可选,自定义的 fetch 方法。 -
getPublicPath -
(entry: Entry) => string
- 可选,参数是微应用的 entry 值。 -
getTemplate -
(tpl: string) => string
- 可选。 -
excludeAssetFilter -
(assetUrl: string) => boolean
- 可选,指定部分特殊的动态加载的微应用资源(css/js) 不被 legions 劫持处理。
-
-
-
用法
启动 qiankun。
-
示例
import { start } from 'legions-micro-service'; start();
-
参数
- appLink -
string
- 必选
- appLink -
-
用法
设置主应用启动后默认进入的微应用。
-
示例
import { setDefaultMountApp } from 'legions-micro-service'; setDefaultMountApp('/homeApp');
-
参数
- effect -
() => void
- 必选
- effect -
-
用法
第一个微应用 mount 后需要调用的方法,比如开启一些监控或者埋点脚本。
-
示例
import { runAfterFirstMounted } from 'legions-micro-service'; runAfterFirstMounted(() => startMonitor());
适用于需要手动 加载/卸载 一个微应用的场景。
通常这种场景下微应用是一个不带路由的可独立运行的业务组件。 微应用不宜拆分过细,建议按照业务域来做拆分。业务关联紧密的功能单元应该做成一个微应用,反之关联不紧密的可以考虑拆分成多个微应用。 一个判断业务关联是否紧密的标准:看这个微应用与其他微应用是否有频繁的通信需求。如果有可能说明这两个微应用本身就是服务于同一个业务场景,合并成一个微应用可能会更合适。-
参数
-
app -
LoadableApp
- 必选,微应用的基础信息- name -
string
- 必选,微应用的名称,微应用之间必须确保唯一。 - entry -
string | { scripts?: string[]; styles?: string[]; html?: string }
- 必选,微应用的入口(详细说明同上)。 - container -
string | HTMLElement
- 必选,微应用的容器节点的选择器或者 Element 实例。如container: '#root'
或container: document.querySelector('#root')
。 - props -
object
- 可选,初始化时需要传递给微应用的数据。
- name -
-
configuration -
Configuration
- 可选,微应用的配置信息-
sandbox -
boolean
|{ strictStyleIsolation?: boolean, experimentalStyleIsolation?: boolean }
- 可选,是否开启沙箱,默认为true
。默认情况下沙箱可以确保单实例场景子应用之间的样式隔离,但是无法确保主应用跟子应用、或者多实例场景的子应用样式隔离。当配置为
基于 ShadowDOM 的严格样式隔离并不是一个可以无脑使用的方案,大部分情况下都需要接入应用做一些适配后才能正常在 ShadowDOM 中运行起来(比如 react 场景下需要解决这些 问题,使用者需要清楚开启了{ strictStyleIsolation: true }
时表示开启严格的样式隔离模式。这种模式下 legions 会为每个微应用的容器包裹上一个 shadow dom 节点,从而确保微应用的样式不会对全局造成影响。strictStyleIsolation
意味着什么。后续 legions 会提供更多官方实践文档帮助用户能快速的将应用改造成可以运行在 ShadowDOM 环境的微应用。除此以外,qiankun 还提供了一个实验性的样式隔离特性,当 experimentalStyleIsolation 被设置为 true 时,qiankun 会改写子应用所添加的样式为所有样式规则增加一个特殊的选择器规则来限定其影响范围,因此改写后的代码会表达类似为如下结构:
// 假设应用名是 react16 .app-main { font-size: 14px; } div[data-legions-react16] .app-main { font-size: 14px; }
注意事项: 目前 @keyframes, @font-face, @import, @page 等规则不会支持 (i.e. 不会被改写)
-
singular -
boolean | ((app: RegistrableApp<any>) => Promise<boolean>);
- 可选,是否为单实例场景,单实例指的是同一时间只会渲染一个微应用。默认为false
。 -
fetch -
Function
- 可选,自定义的 fetch 方法。 -
getPublicPath -
(entry: Entry) => string
- 可选,参数是微应用的 entry 值。 -
getTemplate -
(tpl: string) => string
- 可选 -
excludeAssetFilter -
(assetUrl: string) => boolean
- 可选,指定部分特殊的动态加载的微应用资源(css/js) 不被qiankun 劫持处理
-
-
-
返回值 -
MicroApp
- 微应用实例- mount(): Promise<null>;
- unmount(): Promise<null>;
- update(customProps: object): Promise<any>;
- getStatus(): | "NOT_LOADED" | "LOADING_SOURCE_CODE" | "NOT_BOOTSTRAPPED" | "BOOTSTRAPPING" | "NOT_MOUNTED" | "MOUNTING" | "MOUNTED" | "UPDATING" | "UNMOUNTING" | "UNLOADING" | "SKIP_BECAUSE_BROKEN" | "LOAD_ERROR";
- loadPromise: Promise<null>;
- bootstrapPromise: Promise<null>;
- mountPromise: Promise<null>;
- unmountPromise: Promise<null>;
-
用法
手动加载一个微应用。
如果需要能支持主应用手动 update 微应用,需要微应用 entry 再多导出一个 update 钩子:
export async function mount(props) { renderApp(props); } // 增加 update 钩子以便主应用手动更新微应用 export async function update(props) { renderPatch(props); }
-
示例
import { loadMicroApp } from 'legions-micro-service'; import React from 'react'; class App extends React.Component { containerRef = React.createRef(); microApp = null; componentDidMount() { this.microApp = loadMicroApp( { name: 'app1', entry: '//localhost:1234', container: this.containerRef.current, props: { name: 'legions' } }, ); } componentWillUnmount() { this.microApp.unmount(); } componentDidUpdate() { this.microApp.update({ name: 'kuitos' }); } render() { return <div ref={this.containerRef}></div>; } }
-
参数
- apps -
AppMetadata[]
- 必选 - 预加载的应用列表 - importEntryOpts - 可选 - 加载配置
- apps -
-
类型
-
AppMetadata
- name -
string
- 必选 - 应用名 - entry -
string | { scripts?: string[]; styles?: string[]; html?: string }
- 必选,微应用的 entry 地址
- name -
-
-
用法
手动预加载指定的微应用静态资源。仅手动加载微应用场景需要,基于路由自动激活场景直接配置
prefetch
属性即可。 -
示例
import { prefetchApps } from 'legions-micro-service'; prefetchApps([ { name: 'app1', entry: '//locahost:7001' }, { name: 'app2', entry: '//locahost:7002' } ])
-
参数
- handler -
(...args: any[]) => void
- 必选
- handler -
-
用法
添加全局的未捕获异常处理器。
-
示例
import { addGlobalUncaughtErrorHandler } from 'legions-micro-service'; addGlobalUncaughtErrorHandler(event => console.log(event));
-
参数
- handler -
(...args: any[]) => void
- 必选
- handler -
-
用法
移除全局的未捕获异常处理器。
-
示例
import { removeGlobalUncaughtErrorHandler } from 'legions-micro-service'; removeGlobalUncaughtErrorHandler(handler);
-
参数
- state -
Record<string, any>
- 必选
- state -
-
用法
定义全局状态,并返回通信方法,建议在主应用使用,微应用通过 props 获取通信方法。
-
返回
-
MicroAppStateActions
-
onGlobalStateChange:
(callback: OnGlobalStateChangeCallback, fireImmediately?: boolean) => void
, 在当前应用监听全局状态,有变更触发 callback,fireImmediately = true 立即触发 callback -
setGlobalState:
(state: Record<string, any>) => boolean
, 按一级属性设置全局状态,微应用中只能修改已存在的一级属性 -
offGlobalStateChange:
() => boolean
,移除当前应用的状态监听,微应用 umount 时会默认调用
-
-
-
示例
主应用:
import { initGlobalState, MicroAppStateActions } from 'legions-micro-service'; // 初始化 state const actions: MicroAppStateActions = initGlobalState(state); actions.onGlobalStateChange((state, prev) => { // state: 变更后的状态; prev 变更前的状态 console.log(state, prev); }); actions.setGlobalState(state); actions.offGlobalStateChange();
微应用:
// 从生命周期 mount 中获取通信方法,使用方式和 master 一致 export function mount(props) { props.onGlobalStateChange((state, prev) => { // state: 变更后的状态; prev 变更前的状态 console.log(state, prev); }); props.setGlobalState(state); }