Webpack模块联邦:微前端架构的新选择

Webpack模块联邦(Module Federation)是Webpack 5引入的一项革命性特性,它彻底改变了微前端架构的实现方式。模块联邦允许不同的Web应用程序(或微前端应用)在运行时动态共享代码,无需传统的打包或发布过程中的物理共享。这意味着每个微应用可以独立开发、构建和部署,同时还能轻松地共享组件、库甚至是业务逻辑。

基础概念

  • 容器应用(Container):作为微前端架构的宿主,负责加载和协调各个微应用。
  • 远程应用(Remote):独立的微应用,可以暴露自己的模块给其他应用使用,也可以消费来自其他应用的模块。

实现步骤

1. 容器应用配置

在容器应用的webpack.config.js中,使用ModuleFederationPlugin来声明远程微应用的来源。

// webpack.config.js (Container)
const ModuleFederationPlugin = require('webpack/lib/container/ModuleFederationPlugin');module.exports = {// ...其他配置plugins: [new ModuleFederationPlugin({name: 'container',remotes: {app1: 'app1@http://localhost:3001/remoteEntry.js',app2: 'app2@http://localhost:3002/remoteEntry.js',},shared: {react: { singleton: true },'react-dom': { singleton: true },},}),],
};

这里,remotes字段指定了远程微应用的名称和其远程入口文件URL。shared配置则指明了哪些模块应该作为单例共享,比如React和ReactDOM,以避免重复加载。

2. 远程应用配置

在每个远程应用的webpack.config.js中,同样使用ModuleFederationPlugin,但这次是来暴露自己的模块。

// webpack.config.js (Remote App1)
const ModuleFederationPlugin = require('webpack/lib/container/ModuleFederationPlugin');module.exports = {// ...其他配置plugins: [new ModuleFederationPlugin({name: 'app1',filename: 'remoteEntry.js',exposes: {'./MyComponent': './src/components/MyComponent',},shared: {react: { singleton: true },'react-dom': { singleton: true },},}),],
};

exposes字段定义了哪些模块将对外暴露。在这个例子中,MyComponent组件可以从容器应用或其他微应用中被导入和使用。

3. 消费远程模块

在容器应用或另一个远程应用中,可以直接导入远程暴露的模块。

// In a component of Container or another Remote App
import MyComponent from 'app1/MyComponent';function App() {return (<div><h1>Container App</h1><MyComponent /></div>);
}export default App;

优势

  • 独立开发和部署:每个微应用可以独立开发、构建和部署,提高了开发效率和部署灵活性。
  • 按需加载:只有当某个模块真正被使用时,才会加载对应的远程代码,优化了首屏加载时间和整体性能。
  • 版本管理和隔离:每个微应用可以自由升级其依赖,避免了版本冲突问题。
  • 易于维护和扩展:模块联邦的松耦合特性使得添加或移除微应用变得简单快捷。

Webpack模块联邦通过简化微前端架构中的代码共享机制,为现代Web应用的开发和维护提供了一种高效且灵活的解决方案。

实战案例:构建一个简单的微前端应用

让我们通过一个简单的例子来演示如何使用Webpack模块联邦构建两个微应用:一个容器应用和一个远程应用。

1. 创建容器应用

首先,创建一个新的React应用作为容器应用:

npx create-react-app container-app
cd container-app

安装webpack和webpack-cli(注意,由于create-react-app内部已包含Webpack,通常不需要单独安装,这里仅为演示目的):

npm install webpack webpack-cli --save-dev

修改package.json,增加一个启动脚本来配置Webpack:

"scripts": {"start": "webpack serve --config webpack.config.js",// ...
}

创建webpack.config.js,配置Module Federation Plugin

// webpack.config.js (Container App)
const HtmlWebpackPlugin = require('html-webpack-plugin');
const ModuleFederationPlugin = require('webpack/lib/container/ModuleFederationPlugin');module.exports = {// ...其他配置plugins: [new HtmlWebpackPlugin({ template: './public/index.html' }),new ModuleFederationPlugin({name: 'containerApp',remotes: {remoteApp: 'remoteApp@http://localhost:3010/remoteEntry.js',},shared: {react: { singleton: true },'react-dom': { singleton: true },},}),],
};
2. 创建远程应用

在另一个目录中创建远程应用:

npx create-react-app remote-app
cd remote-app

同样,修改package.json,增加启动脚本,并安装webpackwebpack-cli(仅作示例):

npm install webpack webpack-cli --save-dev

remote-appwebpack.config.js中配置Module Federation Plugin以暴露组件:

// webpack.config.js (Remote App)
const ModuleFederationPlugin = require('webpack/lib/container/ModuleFederationPlugin');module.exports = {// ...其他配置plugins: [new ModuleFederationPlugin({name: 'remoteApp',filename: 'remoteEntry.js',exposes: {'./MyWidget': './src/MyWidget',},shared: {react: { singleton: true },'react-dom': { singleton: true },},}),],
};

remote-app/src目录下创建MyWidget.js组件:

// MyWidget.js
import React from 'react';const MyWidget = () => {return <h1>Hello from Remote App!</h1>;
};export default MyWidget;
3. 容器应用消费远程组件

回到container-app,在需要的地方导入远程组件:

// container-app/src/App.js
import React from 'react';
import MyWidget from 'remoteApp/MyWidget';function App() {return (<div className="App"><header className="App-header"><MyWidget /></header></div>);
}export default App;
4. 启动应用

分别启动两个应用:

# 在远程应用目录
npm start --port 3010
# 在容器应用目录
npm start

现在,在浏览器中访问容器应用,你应该能看到来自远程应用的组件被成功加载和显示。

高级用法和最佳实践

1. 动态加载和懒加载

在实际项目中,你可能希望根据用户的行为或特定条件动态加载远程应用。Webpack模块联邦支持异步加载,只需在导入时使用import()函数即可。

// container-app/src/App.js
import React, { lazy, Suspense } from 'react';
const MyWidget = lazy(() => import('remoteApp/MyWidget'));function App() {return (<div className="App"><Suspense fallback={<div>Loading...</div>}><MyWidget /></Suspense></div>);
}export default App;

这样,MyWidget组件将在需要时按需加载,提高首屏加载速度。

2. 版本管理和依赖管理

在微前端架构中,确保不同应用之间的依赖版本兼容是关键。使用ModuleFederationPluginshared配置,你可以指定共享模块的版本范围和加载策略(例如,singletonstrictVersion等)。

javascript
// webpack.config.js
new ModuleFederationPlugin({// ...shared: {react: { version: '^17.0.0', singleton: true },'react-dom': { version: '^17.0.0', singleton: true },},
}),
3. 路由集成

在微前端架构中,路由管理是一个重要的组成部分。你可以使用像react-router-dom这样的库,结合Microfrontends-Router或自定义解决方案来实现跨应用的路由跳转。

// container-app/src/Routes.js
import { BrowserRouter as Router, Route, Switch } from 'react-router-dom';
import App1 from './App1';
import App2 from './App2';function Routes() {return (<Router><Switch><Route path="/app1" component={App1} /><Route path="/app2" component={App2} /></Switch></Router>);
}export default Routes;
4. 状态管理

对于共享状态的需求,可以使用Redux、MobX或Context API等状态管理库,或者专门针对微前端设计的状态管理库如single-spa-reduxqiankunstore解决方案等。

5. 共享服务和公共库

除了组件外,你还可以共享服务和公共库。例如,创建一个专门的远程应用来提供API服务,或者共享一个公共的HTTP库。

// webpack.config.js (Remote App for Services)
new ModuleFederationPlugin({name: 'services',filename: 'remoteEntry.js',exposes: {'./ApiService': './src/services/ApiService','./HttpLibrary': './src/libs/http-library',},shared: {// ...其他共享库},
}),
6. 错误处理和日志记录

为了确保微前端应用的稳定运行,需要实现全局错误捕获和日志记录。可以使用window.onerrortry...catch语句,或者使用专门的日志库如log4js。

// container-app/src/index.js
window.onerror = function (errorMessage, fileName, lineNumber, columnNumber, error) {// 记录错误信息console.error(errorMessage, fileName, lineNumber, columnNumber, error);// ...其他处理逻辑return true; // 阻止浏览器默认错误处理
};

2024年礼包2500G计算机入门到高级架构师开发资料超级大礼包免费送!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/10591.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

javax.net.ssl.SSLException: Received fatal alert: protocol_version已经解决

起因&#xff1a; 在帮别人讲解项目时&#xff0c;将项目的tomcat配置完&#xff0c;点击运行后&#xff0c;报错&#xff0c;信息如标题。 解决办法&#xff1a; 在csdn百度问题&#xff0c;得到的方法主要有几个&#xff1a; 1.jdk要配置在1.8以上&#xff1b; 2.数据库地…

Stable Diffusion是什么?

目录 一、Stable Diffusion是什么&#xff1f; 二、Stable Diffusion的基本原理 三、Stable Diffusion有哪些运用领域&#xff1f; 一、Stable Diffusion是什么&#xff1f; Stable Diffusion是一个先进的人工智能图像生成模型&#xff0c;它能够根据文本描述创造出高质量的图…

Spring IOC容器加载流程原理源码深度剖析

Spring IOC容器加载流程原理源码深度剖析 介绍 在Java开发中&#xff0c;Spring框架是非常常用的一个框架&#xff0c;它提供了很多便捷的功能和特性&#xff0c;其中最重要的就是IOC&#xff08;Inversion of Control&#xff09;容器。Spring IOC容器是Spring框架的核心&am…

邻接矩阵基础入门

引言 邻接矩阵是图论中表示图的一种方式&#xff0c;它通过矩阵来描述图中各顶点之间的连接关系。在邻接矩阵中&#xff0c;图中的每个顶点都对应矩阵中的一行和一列&#xff0c;矩阵中的元素表示顶点之间是否存在边以及边的权重&#xff08;如果是加权图&#xff09;。 定义和…

什么是ip地址?和dns地址的区别是什么?

大家都知道电脑想要上网需要配置ip地址和dns地址&#xff0c;手动配置、自动获取均可&#xff0c;但是ip地址是什么&#xff1f;和dns又有什么区别呢&#xff1f;小编给大家详细解释一下。 IP地址 IP,就是Internet Protocol的缩写&#xff0c;是一种通信协议&#xff0c;我们用…

信息检索(36):ConTextual Masked Auto-Encoder for Dense Passage Retrieval

ConTextual Masked Auto-Encoder for Dense Passage Retrieval 标题摘要1 引言2 相关工作3 方法3.1 初步&#xff1a;屏蔽自动编码3.2 CoT-MAE&#xff1a;上下文屏蔽自动编码器3.3 密集通道检索的微调 4 实验4.1 预训练4.2 微调4.3 主要结果 5 分析5.1 与蒸馏检索器的比较5.2 …

Flink面试整理-了解如何对Flink进行扩展和定制化开发,以满足特定的业务需求

Apache Flink 是一个灵活且可扩展的流处理框架,允许进行定制化开发以满足特定的业务需求。以下是对 Flink 进行扩展和定制化开发的几个主要方向: 1. 自定义算子(Operators) Flink 允许开发者编写自定义算子来实现特定的数据处理逻辑。例如,可以通过扩展 RichMapFunction 或…

AI学习指南概率论篇-条件概率和独立性

AI学习指南概率论篇-条件概率和独立性 概述 在人工智能领域&#xff0c;概率论是一项至关重要的工具。概率论不仅仅是用于量化不确定性&#xff0c;还在AI系统中扮演着关键角色。其中&#xff0c;条件概率和独立性是概率论中重要的概念之一&#xff0c;它们在AI领域有着广泛的…

适配qnx和linux平台的线程管理类封装

概述 封装代码仓库&#xff1a; https://gitee.com/liudegui/my_thread 尝试封装一个基于C11的多线程控制与调度类&#xff0c;适配QNX和Linux平台&#xff0c;它提供了以下主要功能&#xff1a; 线程的创建与销毁管理。线程的优先级调度。线程的CPU亲和性设置。线程的等待与…

[大师C语言(第四篇)]C语言段错误原理研究

C语言段错误原理研究&#xff08;一&#xff09; 段错误&#xff08;Segmentation Fault&#xff09;是C语言程序中常见的错误类型&#xff0c;它通常发生在程序尝试访问非法内存区域时。本文将深入探讨C语言段错误的原理&#xff0c;并分析其背后的技术原理。 段错误的定义 …

matlab人脸识别

在MATLAB中实现人脸识别通常涉及到图像处理、特征提取和分类器的使用。下面是一个简化的MATLAB人脸识别代码的概述&#xff0c;使用了PCA&#xff08;主成分分析&#xff09;作为特征提取方法&#xff0c;以及简单的分类器&#xff08;如最近邻分类器&#xff09;进行分类。请注…

无障碍Web开发:遵循WCAG标准构建包容性用户体验

无障碍Web开发旨在确保所有用户&#xff0c;无论其身体条件或能力如何&#xff0c;都能轻松、有效地访问和使用Web内容。遵循Web Content Accessibility Guidelines (WCAG) 标准是实现这一目标的关键。以下是一些基于WCAG标准的无障碍Web开发实践&#xff0c;以构建更具包容性的…

2024数维杯要点和难点,具体案例

2024数维杯&#xff0c;全称为2024年第九届数维杯大学生数学建模挑战赛&#xff0c;是由内蒙古创新教育学会主办的一项数学建模竞赛。该竞赛旨在培养学生的创新意识及运用数学方法和计算机技术解决实际问题的能力。以下是关于2024数维杯的一些关键信息&#xff1a; 竞赛时间&am…

Django 从零到一:Django环境设置

文章目录 安装 Python 3.11.0创建 Python 虚拟环境激活虚拟环境退出虚拟环境 配置 pip 国内源安装 Django 4.2本章小结 常言道&#xff1a;“工欲善其事&#xff0c;必先利其器”。我们先设置一下需要的环境。 我们使用的软件如下&#xff1a; Python 3.11.0Django 4.2Django…

UNXIU

外设可以对程序存储器和选项字节进行擦除和编程&#xff0c;不能对系统存储器进行操作&#xff0c;因为系统存储器是原厂写入的bootloader程序&#xff0c;不允许修改 对于C8T6程序存储容量是64K&#xff0c;一般写一个程序只占前边很小一部分空间&#xff0c;剩下的大部分空间…

网络运维故障排错思路!!!!!(稳了!!!)

1 网络排错的必备条件 为什么要先讲必备条件&#xff1f;因为这里所讲的网络排错并不仅仅是停留在某一个小小命令的使用上&#xff0c;而是一套系统的方法&#xff0c;如果没有这些条件&#xff0c;我真的不能保证下面讲的这些你可以听得懂&#xff0c;并且能运用到实际当中&a…

面试 Java 并发编程八股文十问十答第十五期

面试 Java 并发编程八股文十问十答第十五期 作者&#xff1a;程序员小白条&#xff0c;个人博客 相信看了本文后&#xff0c;对你的面试是有一定帮助的&#xff01;关注专栏后就能收到持续更新&#xff01; ⭐点赞⭐收藏⭐不迷路&#xff01;⭐ 1&#xff09;什么是锁的自适应…

Kubernetes 控制平面的安全管理

目录 1. API Server 安全2. etcd 安全3. 网络策略4. 日志与审计5. 定期安全检查与更新6. 云提供商安全集成 Kubernetes 控制平面的安全管理是维护整个集群稳定性和保护敏感信息的关键。控制平面主要包括 API Server、etcd、Controller Manager 和 Scheduler 组件。 1. API Ser…

高斯-牛顿法C实现

高斯-牛顿法(Gauss-Newton method)是一种用于求解非线性最小二乘问题的迭代优化算法。其核心思想是通过近似二阶泰勒展开来简化求解过程,并利用雅可比矩阵(Jacobian matrix)来更新迭代方向。 下面是一个高斯-牛顿法的简单C语言实现示例。这个示例假定我们有一个非线性最小…

Python模块之Numpy(一)-- 创建数组

Numpy是Python用于数据科学计算的基础模块&#xff0c;NumPy 的数据容器能够保存任意类型的数据&#xff0c;可以无缝快速整合各种数据&#xff0c;有助于更加高效地使用pandas等数据处理工具。 数组操作 以下代码是创建一维数组与多维数组并查看数组属性的过程&#xff1a; i…