在正式介绍 qiankun 之前,我们需要知道,qiankun 是一个基于 single-spa 的微前端实现库,旨在帮助大家能更简单、无痛的构建一个生产可用微前端架构系统。
微前端的概念借鉴自后端的微服务,主要是为了解决大型工程在变更、维护、扩展等方面的困难而提出的。目前主流的微前端方案包括以下几个:
- iframe
- 基座模式,主要基于路由分发,qiankun 和 single-spa 就是基于这种模式
- 组合式集成,即单独构建组件,按需加载,类似 npm 包的形式
- EMP,主要基于 Webpack5 Module Federation
- Web Components
严格来讲,这些方案都不算是完整的微前端解决方案,它们只是用于解决微前端中运行时容器的相关问题。
本文我们主要对 qiankun 所基于的基座模式进行介绍。它的主要思路是将一个大型应用拆分成若干个更小、更简单,可以独立开发、测试和部署的微应用,然后由一个基座应用根据路由进行应用切换。
-
🥄 简单
由于主应用微应用都能做到技术栈无关,qiankun 对于用户而言只是一个类似 jQuery 的库,你需要调用几个 qiankun 的 API 即可完成应用的微前端改造。同时由于 qiankun 的 HTML entry 及沙箱的设计,使得微应用的接入像使用 iframe 一样简单。
-
🍡 解耦/技术栈无关
微前端的核心目标是将巨石应用拆解成若干可以自治的松耦合微应用,而 qiankun 的诸多设计均是秉持这一原则,如 HTML entry、沙箱、应用间通信等。这样才能确保微应用真正具备 独立开发、独立运行 的能力。
- 📦 基于 single-spa 封装,提供了更加开箱即用的 API。
- 📱 技术栈无关,任意技术栈的应用均可 使用/接入,不论是 React/Vue/Angular/JQuery 还是其他等框架。
- 💪 HTML Entry 接入方式,让你接入微应用像使用 iframe 一样简单。
- 🛡 样式隔离,确保微应用之间样式互相不干扰。
- 🧳 JS 沙箱,确保微应用之间 全局变量/事件 不冲突。
- ⚡️ 资源预加载,在浏览器空闲时间预加载未打开的微应用资源,加速微应用打开速度。
- 🔌 umi 插件,提供了 @umijs/plugin-qiankun 供 umi 应用一键切换成微前端架构系统。
本文适合刚接触
qiankun
的新人,介绍了如何从 0 构建一个qiankun
项目。项目主要有以下构成:
-
主应用:
- 使用 umi3.5,未使用 @umijs/plugin-qiankun,而是直接使用的 qiankun
-
vue 微应用:
- 使用 vue2.x 创建
- 使用 vue3.x,暂未使用 vite 构建,目测 vite 不兼容
-
react 微应用:
- 使用 create-react-app 创建
-
umi3 微应用:
- 使用 umi3.结合插件 @umijs/plugin-qiankun
-
非 webpack 构建的微应用:
- 一些非 webpack 构建的项目,例如 jQuery 项目、jsp 项目,都可以按照这个处理。
- 接入之前请确保你的项目里的图片、音视频等资源能正常加载,如果这些资源的地址都是完整路径(例如 https://qiankun.umijs.org/logo.png ),则没问题。如果都是相对路径,需要先将这些资源上传到服务器,使用完整路径。
-
Angular 微应用:
- 使用的 @angular/[email protected] 版本
主应用按照官方的说法,不限技术栈,只需要提供一个容器 DOM,然后注册微应用并 start 即可。这里我们使用 umi 来初始化。
# 项目初始化
$ yarn create @umijs/umi-app
# 安装依赖
$ yarn
# 启动
$ yarn start
# 安装 qiankun
$ yarn add qiankun
基本环境搭建完成,在主应用中增加一些菜单和路由,用于主应用页面以及主应用和微应用之间切换操作。页面布局和路由配置这里不做过多介绍,文末会奉上源码。大致页面如下图:
注册微应用的基础配置信息。当浏览器 url 发生变化时,会自动检查每一个微应用注册的 activeRule 规则,符合规则的应用将会被自动激活。本示列分别有一个主应用五个微应用构成,在主应用中增加微应用的配置文件,对注册微应用做单独的管理。
主应用 src 文件下增加
registerMicroAppsConfig.ts
,内容如下:
const loader = (loading: boolean) => {
// 此处可以获取微应用是否加载成功,可以用来触发全局的 loading
console.log("loading", loading);
};
export const Microconfig = [
//name: 微应用的名称,
//entry: 微应用的入口,
//container: 微应用的容器节点的选择器或者 Element 实例,
//activeRule: 激活微应用的规则(可以匹配到微应用的路由),
//loader: 加载微应用的状态 true | false
{
name: "vue2",
entry: "http://localhost:8001",
container: "#subContainer",
activeRule: "/vue2",
loader,
},
{
name: "vue3",
entry: "http://localhost:8002",
container: "#subContainer",
activeRule: "/vue3",
loader,
},
{
name: "react",
entry: "http://localhost:8003",
container: "#subContainer",
activeRule: "/react",
loader,
},
{
name: "umi",
entry: "http://localhost:8004",
container: "#subContainer",
activeRule: "/umi",
loader,
},
{
name: "purehtml",
entry: "http://127.0.0.1:8005",
container: "#subContainer",
activeRule: "/purehtml",
loader,
},
//angular
{
name: "angular",
entry: "http://127.0.0.1:8006",
container: "#subContainer",
activeRule: "/angular",
loader,
},
];
主应用入口文件引入(主应用使用的 umi,所以直接在 pages/index.tsx 引入)
import LayoutPage from "@/layout/index";
import {
registerMicroApps,
start,
addGlobalUncaughtErrorHandler,
} from "qiankun";
import { Microconfig } from "@/registerMicroAppsConfig";
// 注册微应用
registerMicroApps(Microconfig, {
// qiankun 生命周期钩子 - 微应用加载前
beforeLoad: (app: any) => {
console.log("before load", app.name);
return Promise.resolve();
},
// qiankun 生命周期钩子 - 微应用挂载后
afterMount: (app: any) => {
console.log("after mount", app.name);
return Promise.resolve();
},
});
// 启动 qiankun
start();
export default function IndexPage({ children }: any) {
return (
<LayoutPage>
<div>{children}</div>
{/* 增加容器,用于显示微应用 */}
<div id="subContainer"></div>
</LayoutPage>
);
}
// 添加全局异常捕获
addGlobalUncaughtErrorHandler((handler) => {
console.log("异常捕获", handler);
});
- ⚡️prefetch: 开启预加载
- true | 'all' | string[] | function
- 🧳sandbox:是否开启沙箱
- strictStyleIsolation 严格模式(
ShadowDOM
) - experimentalStyleIsolation 实验性方案,建议使用
- strictStyleIsolation 严格模式(
start({
prefetch: true, // 开启预加载
sandbox: {
experimentalStyleIsolation: true, // 开启沙箱模式,实验性方案
},
});
import { setDefaultMountApp } from "qiankun"
setDefaultMountApp('/purehtml');
注意微应用的名称
package.json
=>name
需要和主应用中注册时的name
相对应,且必须确保唯一。
# 安装 vueCli
$ yarn add @vue/cli
# 创建项目
$ vue create vue2.x_root
# 选择 vue2 版本
# 安装依赖
$ yarn
# 启动
$ yarn serve
- 在
src
目录新增public-path.js
:
if (window.__POWERED_BY_QIANKUN__) {
__webpack_public_path__ = window.__INJECTED_PUBLIC_PATH_BY_QIANKUN__;
}
- 入口文件
main.js
修改
import "./public-path";
import Vue from "vue";
import App from "./App.vue";
import VueRouter from "vue-router";
import routes from "./router";
Vue.config.productionTip = false;
let router = null;
let instance = null;
function render(props = {}) {
const { container } = props;
router = new VueRouter({
// 注意这里的name,最好不要写死,直接使用主应用传过来的name
base: window.__POWERED_BY_QIANKUN__ ? `${props.name}` : "/",
mode: "history",
routes,
});
Vue.use(VueRouter);
instance = new Vue({
router,
render: (h) => h(App),
}).$mount(container ? container.querySelector("#app") : "#app");
}
// 独立运行时
if (!window.__POWERED_BY_QIANKUN__) {
render();
}
export async function bootstrap() {
console.log("[vue2] vue app bootstraped");
}
export async function mount(props) {
render(props);
}
export async function unmount() {
instance.$destroy();
instance.$el.innerHTML = "";
instance = null;
router = null;
}
- 打包配置修改(
vue.config.js
):
const path = require("path");
const { name } = require("./package");
function resolve(dir) {
return path.join(__dirname, dir);
}
module.exports = {
filenameHashing: true,
lintOnSave: process.env.NODE * ENV !== "production",
runtimeCompiler: true,
productionSourceMap: false,
devServer: {
hot: true,
disableHostCheck: true,
// 修改默认端口,和注册时一直
port: 8001,
overlay: {
warnings: false,
errors: true,
},
// 解决主应用加载子应用出现跨域问题
headers: {
"Access-Control-Allow-Origin": "*",
},
},
// 自定义 webpack 配置
configureWebpack: {
resolve: {
alias: {
"@": resolve("src"),
},
},
// 让主应用能正确识别微应用暴露出来的一些信息
output: {
library: `${name}-[name]`,
libraryTarget: "umd", // 把子应用打包成 umd 库格式
jsonpFunction: `webpackJsonp*${name}`,
},
},
};
- 主应用查看加载效果
# 安装 vueCli
$ yarn add @vue/cli
# 创建项目
$ vue create vue3.x_root
# 选择 vue3 版本
# 安装依赖
$ yarn
# 启动
$ yarn serve
- 在
src
目录新增public-path.js
:
if (window.__POWERED_BY_QIANKUN__) {
__webpack_public_path__ = window.__INJECTED_PUBLIC_PATH_BY_QIANKUN__;
}
- 入口文件
main.ts
修改
//@ts-nocheck
import './public-path';
import { createApp } from 'vue';
import { createRouter, createWebHistory } from 'vue-router';
import App from './App.vue';
import routes from './router';
import store from './store';
let router = null;
let instance = null;
let history = null;
function render(props = {}) {
const { container } = props;
history = createWebHistory(window.__POWERED_BY_QIANKUN__ ? `${props.name}` : '/');
router = createRouter({
history,
routes,
});
instance = createApp(App);
instance.use(router);
instance.use(store);
instance.mount(container ? container.querySelector('#app') : '#app');
}
if (!window.__POWERED_BY_QIANKUN__) {
render();
}
export async function bootstrap() {
console.log('%c ', 'color: green;', 'vue3.0 app bootstraped');
}
export async function mount(props) {
render(props);
}
export async function unmount() {
instance.unmount();
instance._container.innerHTML = '';
instance = null;
router = null;
history.destroy();
}
- 打包配置修改(
vue.config.js
):
const path = require('path')
const { name } = require('./package')
function resolve (dir) {
return path.join(__dirname, dir)
}
module.exports = {
filenameHashing: true,
lintOnSave: process.env.NODE_ENV !== 'production',
runtimeCompiler: true,
productionSourceMap: false,
devServer: {
hot: true,
disableHostCheck: true,
// 修改默认端口,和注册时一直
port: 8002,
overlay: {
warnings: false,
errors: true
},
headers: {
'Access-Control-Allow-Origin': '*'
}
},
// 自定义webpack配置
configureWebpack: {
resolve: {
alias: {
'@': resolve('src')
}
},
// 让主应用能正确识别微应用暴露出来的一些信息
output: {
library: `${name}-[name]`,
libraryTarget: 'umd', // 把子应用打包成 umd 库格式
jsonpFunction: `webpackJsonp_${name}`
}
}
}
- 主应用查看加载效果
# 创建项目
$ yarn add create-react-app react_root
# 启动
$ yarn start
- 在
src
目录新增public-path.js
:
if (window.__POWERED_BY_QIANKUN__) {
__webpack_public_path__ = window.__INJECTED_PUBLIC_PATH_BY_QIANKUN__;
}
- 设置 history 模式路由的 base:
刚刚创建的项目没有路由,所以先要安装路由
# 路由安装
$ yarn add react-router react-router-dom
入口文件 index.js 修改,为了避免根 id #root 与其他的 DOM 冲突,需要限制查找范围。
import './public-path';
import React from 'react';
import ReactDOM from 'react-dom';
import App from './App';
import { BrowserRouter, Route, Link } from "react-router-dom"
function render(props) {
const { container } = props;
ReactDOM.render(
<BrowserRouter basename={window.__POWERED_BY_QIANKUN__ ? '/react' : '/'}>
<App/>
</BrowserRouter>
, container ? container.querySelector('#root') : document.querySelector('#root'));
}
if (!window.__POWERED_BY_QIANKUN__) {
render({});
}
export async function bootstrap() {
console.log('[react16] react app bootstraped');
}
export async function mount(props) {
console.log('[react16] props from main framework', props);
render(props);
}
export async function unmount(props) {
const { container } = props;
ReactDOM.unmountComponentAtNode(container ? container.querySelector('#root') : document.querySelector('#root'));
}
- webpack 打包配置修改
安装插件 @rescripts/cli,当然也可以选择其他的插件,例如 react-app-rewired
# 安装
$ yarn add @rescripts/cli
根目录增加配置文件
.rescriptsrc.js
,注意一定是根目录下哦
const { name } = require('./package');
module.exports = {
webpack: (config) => {
config.output.library = `${name}-[name]`;
config.output.libraryTarget = 'umd';
config.output.jsonpFunction = `webpackJsonp_${name}`;
config.output.globalObject = 'window';
return config;
},
devServer: (_) => {
const config = _;
config.headers = {
'Access-Control-Allow-Origin': '*',
};
config.historyApiFallback = true;
config.hot = false;
config.watchContentBase = false;
config.liveReload = false;
return config;
},
};
package.json
配置修改
{
"name": "react_root",
"version": "0.1.0",
"private": true,
"dependencies": {
"@rescripts/cli": "^0.0.16",
"@testing-library/jest-dom": "^5.11.4",
"@testing-library/react": "^11.1.0",
"@testing-library/user-event": "^12.1.10",
"react": "^17.0.2",
"react-dom": "^17.0.2",
"react-router-dom": "5.0",
"react-scripts": "4.0.3",
"web-vitals": "^1.0.1"
},
"scripts": {
"start": "set PORT=8003&&rescripts start",
"build": "rescripts build",
"test": "rescripts test",
"eject": "rescripts eject"
},
"eslintConfig": {
"extends": [
"react-app",
"react-app/jest"
]
},
"browserslist": {
"production": [
">0.2%",
"not dead",
"not op_mini all"
],
"development": [
"last 1 chrome version",
"last 1 firefox version",
"last 1 safari version"
]
}
}
- 主应用查看加载效果
umi 项目初始化方式参考初始化主应用的方式。umi 应用使用
@umijs/plugin-qiankun
可以一键开启微前端模式。
- 安装插件
# 安装 @umijs/plugin-qiankun
$ yarn add @umijs/plugin-qiankun
- 修改配置文件
umirc.ts
如果是配置文件抽离到
config
中,直接修改config.js
import { defineConfig } from 'umi';
export default defineConfig({
nodeModulesTransform: {
type: 'none',
},
routes: [
{ path: '/', component: '@/pages/index' },
],
fastRefresh: {},
//开启qiankun配置
qiankun:{
slave:{
}
}
});
这里只是做了简单的集成配置,更过功能请参看@umijs/plugin-qiankun
- 加载效果
非 webpack 应用有个需要注意点的点:接入之前请确保你的项目里的图片、音视频等资源能正常加载,如果这些资源的地址都是完整路径(例如 https://qiankun.umijs.org/logo.png),则没问题。如果都是相对路径,需要先将这些资源上传到服务器,使用完整路径。
- 入口文件声明
entry
入口
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Document</title>
</head>
<script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.0/jquery.min.js"></script>
<body>
<div id="test">测试微应用</div>
</body>
</html>
<!-- entry 入口 -->
<script src="./index.js" entry></script>
- index.js
const render = ($) => {
// 这里可以在渲染之前做些什么。。。
return Promise.resolve();
};
((global) => {
//purehtml 是对应的微应用名称
global['purehtml'] = {
bootstrap: () => {
console.log('purehtml bootstrap');
return Promise.resolve();
},
mount: (props) => {
console.log('purehtml mount00000000000',props);
props.onGlobalStateChange((state,prev)=>{
console.log(state,prev)
})
return render($);
},
unmount: () => {
console.log('purehtml unmount');
return Promise.resolve();
},
};
})(window);
- 为了方便启动和加载,使用
http-server
启动本地服务根目录增加
package.json
文件, 注意name
:purehtml
{
"name": "purehtml",
"version": "1.0.0",
"description": "",
"main": "index.html",
"scripts": {
"start": "cross-env PORT=8005 http-server . --cors",
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "",
"license": "MIT",
"devDependencies": {
"cross-env": "^7.0.2",
"http-server": "^0.12.1"
}
}
- 加载效果
# 安装 CLI
$ yarn add -g @angular/[email protected]
# 创建项目
$ ng new angular_root
# 启动
$ ng serve
- 在
src
目录新增public-path.js
:
if (window.__POWERED_BY_QIANKUN__) {
// eslint-disable-next-line no-undef
__webpack_public_path__ = window.__INJECTED_PUBLIC_PATH_BY_QIANKUN__;
}
}
- 设置 history 模式路由的 base,
src/app/app-routing.module.ts
文件:
import { NgModule } from '@angular/core';
import { Routes, RouterModule } from '@angular/router';
import { APP_BASE_HREF } from '@angular/common';
const routes: Routes = [];
@NgModule({
imports: [RouterModule.forRoot(routes)],
exports: [RouterModule],
// @ts-ignore
providers: [{ provide: APP_BASE_HREF, useValue: window.__POWERED_BY_QIANKUN__ ? '/angular' : '/' }]
})
export class AppRoutingModule { }
- 修改入口文件,src/main.ts 文件
import './public-path';
import { enableProdMode, NgModuleRef } from '@angular/core';
import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';
import { AppModule } from './app/app.module';
import { environment } from './environments/environment';
if (environment.production) {
enableProdMode();
}
let app: void | NgModuleRef<AppModule>;
async function render() {
app = await platformBrowserDynamic()
.bootstrapModule(AppModule)
.catch((err) => console.error(err));
}
if (!(window as any).__POWERED_BY_QIANKUN__) {
render();
}
export async function bootstrap(props: Object) {
console.log(props);
}
export async function mount(props: Object) {
render();
}
export async function unmount(props: Object) {
console.log(props);
// @ts-ignore
app.destroy();
}
- 修改 webpack 打包配置
根据官方指示:先安装
@angular-builders/custom-webpack
,注意:angular 9 项目只能安装 9.x 版本,angular 10 项目可以安装最新版。
$ yarn add @angular-builders/[email protected]
在根目录增加
custom-webpack.config.js
const appName = require('./package.json').name;
module.exports = {
devServer: {
headers: {
'Access-Control-Allow-Origin': '*',
},
},
output: {
library: `${appName}-[name]`,
libraryTarget: 'umd',
jsonpFunction: `webpackJsonp_${appName}`,
},
};
修改 angular.json 配置文件
{
"$schema": "./node_modules/@angular/cli/lib/config/schema.json",
"version": 1,
"newProjectRoot": "projects",
"projects": {
"angularRoot": {
"projectType": "application",
"schematics": {
"@schematics/angular:component": {
"style": "scss"
}
},
"root": "",
"sourceRoot": "src",
"prefix": "app",
"architect": {
"build": {
"builder": "@angular-builders/custom-webpack:browser",
"options": {
"outputPath": "dist/angularRoot",
"index": "src/index.html",
"main": "src/main.ts",
"polyfills": "src/polyfills.ts",
"tsConfig": "tsconfig.app.json",
"aot": true,
"assets": [
"src/favicon.ico",
"src/assets"
],
"styles": [
"src/styles.scss"
],
"scripts": [],
"customWebpackConfig": {
"path": "./custom-webpack.config.js"
}
},
"configurations": {
"production": {
"fileReplacements": [
{
"replace": "src/environments/environment.ts",
"with": "src/environments/environment.prod.ts"
}
],
"optimization": true,
"outputHashing": "all",
"sourceMap": false,
"extractCss": true,
"namedChunks": false,
"extractLicenses": true,
"vendorChunk": false,
"buildOptimizer": true,
"budgets": [
{
"type": "initial",
"maximumWarning": "2mb",
"maximumError": "5mb"
},
{
"type": "anyComponentStyle",
"maximumWarning": "6kb",
"maximumError": "10kb"
}
]
}
}
},
"serve": {
"builder": "@angular-builders/custom-webpack:dev-server",
"options": {
"browserTarget": "angularRoot:build"
},
"configurations": {
"production": {
"browserTarget": "angularRoot:build:production"
}
}
},
"extract-i18n": {
"builder": "@angular-devkit/build-angular:extract-i18n",
"options": {
"browserTarget": "angularRoot:build"
}
},
"test": {
"builder": "@angular-devkit/build-angular:karma",
"options": {
"main": "src/test.ts",
"polyfills": "src/polyfills.ts",
"tsConfig": "tsconfig.spec.json",
"karmaConfig": "karma.conf.js",
"assets": [
"src/favicon.ico",
"src/assets"
],
"styles": [
"src/styles.scss"
],
"scripts": []
}
},
"lint": {
"builder": "@angular-devkit/build-angular:tslint",
"options": {
"tsConfig": [
"tsconfig.app.json",
"tsconfig.spec.json",
"e2e/tsconfig.json"
],
"exclude": [
"**/node_modules/**"
]
}
},
"e2e": {
"builder": "@angular-devkit/build-angular:protractor",
"options": {
"protractorConfig": "e2e/protractor.conf.js",
"devServerTarget": "angularRoot:serve"
},
"configurations": {
"production": {
"devServerTarget": "angularRoot:serve:production"
}
}
}
}
}
},
"defaultProject": "angular"
}
- 启动尝试加载
哇咔咔!!! 报错。。。
-
解决方式
- 主应用中安装
zoom.js
, 并且在import qiankun
之前引入 - 将微应用的
src/polyfills.ts
里面的引入zone.js
- 微应用
src/index.html
<head>
中引入zone.js
- 主应用中安装
- 再次启动尝试加载
哇咔咔!!! 又报错了。。。 什么鬼,页面倒是加载出来了,但是报了一串红
查阅资料,貌似是热更新的
bug
啊。 这里不做过多解释,暴力解决方案:作为子应用时不使用热更新
。
package.json
=>script
中增加如下命令:
"serve:qiankun": "ng serve --disable-host-check --port 8006 --base-href /angular --live-reload false"
作为微应用时使用:
ng serve:qiankuan
启动加载
build 报错问题: 修改
tsconfig.json
文件
{
"compileOnSave": false,
"compilerOptions": {
"baseUrl": "./",
"outDir": "./dist/out-tsc",
"sourceMap": true,
"declaration": false,
"downlevelIteration": true,
"experimentalDecorators": true,
"module": "esnext",
"moduleResolution": "node",
"importHelpers": true,
"target": "es5",
"typeRoots": ["node_modules/@types"],
"lib": ["es2018", "dom"]
},
"angularCompilerOptions": {
"fullTemplateTypeCheck": true,
"strictInjectionParameters": true
}
}
- 查看加载效果
多个应用间通信,这里举个简单的例子:主应用中登录获取用户
id
,当加载微应用时,微应用需要根据不同的用户id
展示不同的数据或者展示不同的页面。这个时候就需要主应用中把对应的用户id
传到微应用中去。传值方式,这里总结了三种方式:
- 挂载微应用时直接
props
传值 initGlobalState
定义全局状态- 定义全局的状态池
注册微应用的基础配置信息时,增加
props
,传入微应用需要的信息
{
name: 'vue2',
entry: 'http://localhost:8001',
container: '#subContainer',
activeRule: '/vue2',
//props
props: {
id: 'props基础传值方式'
},
loader,
}
微应用中在
mount
生命周期props
中获取
export async function mount(props) {
console.log('获取主应用传值',props)
render(props);
}
定义全局状态,并返回通信方法,建议在主应用使用,微应用通过
props
获取通信方法。
- 主应用中声明全局状态
// 全局状态
const state = {
id: 'main_主应用',
};
// 初始化 state
const actions: MicroAppStateActions = initGlobalState(state);
// 监听状态变更
actions.onGlobalStateChange((state, prev) => {
// state: 变更后的状态; prev 变更前的状态
console.log(state, prev);
});
- 微应用获取通信,同样在
mount
生命周期中获取
export async function mount(props) {
console.log('initGlobalState传值',props)
render(props);
}
打印出来发现好像并没有我们需要的值:
我想在这里,细心的同学应该会发现,好像有个
onGlobalStateChange
,setGlobalState
这两个方法,见名知意,应该是用来做状态的监听和修改使用的。不管什么神仙,先调用下试试看喽
封装一个
storeTest
方法做统一调用
function storeTest(props) {
props.onGlobalStateChange &&
props.onGlobalStateChange(
(value, prev) => console.log(`[onGlobalStateChange - ${props.name}]:`, value, prev),
true,
);
// 为了演示效果明显增加定时器
setTimeout(() =>{
props.setGlobalState &&
props.setGlobalState({
id: `${props.name}_子应用`
});
},3000)
}
export async function mount(props) {
storeTest(props);
render(props);
}
输出两次 ???
输出两次的原因是在
微应用
中调用setGlobalState
, 主应用中的onGlobalStateChange
也会执行
- 总结下
initGlobalState
初始化state
onGlobalStateChange
监听状态变更setGlobalState
修改状态offGlobalStateChange
移除监听
- 问题
如果想在微应用某个页面内修改全局状态应该怎么做 ? 当然是可以把
props
中的方法挂载到当前应用的全局上啦。例如:
export async function mount(props) {
storeTest(props);
render(props);
// 挂载到全局 instance 上
instance.config.globalProperties.$onGlobalStateChange = props.onGlobalStateChange;
instance.config.globalProperties.$setGlobalState = props.setGlobalState;
}
定义全局状态池,说白了就是在主应用中定义全局状态,可以使用
redux
vuex
等来定义。定义好全局状态,可以定义一个全局的类,类中声明两个方法,一个用来获取全局状态,一个用来修改全局状态。定义好之后,把这个类通过第一种props
的传值方式传入,微应用通过mount
=>props
接收。这种方式就不做演示,个人建议使用第二种方式。
到这里,基于
qiankun
的微前端搭建基本完成。本文只是对qiankun从0搭建到搭建过程中遇到问题并且解决问题以及后期项目中的一些基础配置和使用做简单概述。下一次将会对多应用部署
问题做个详细概述。