当前位置: 首页 > article >正文

微前端随笔

✨ single-spa:

  1. js-entry
    通过es-module 或 umd 动态插入 js 脚本 ,在主应用中发送请求,来获取子应用的包,
    该子应用的包

    singleSpa.registerApplication({name: 'app1',app: () => import('http://localhost:8080/app1.js'),activeWhen: '/app1'
    });
    

    主应用直接执行 app: () => import('http://localhost:8080/app1.js')该脚本,拿到子应用包。该包中暴露了主应用规定的那几个生命周期函数,

🎨 qiankun:

1. html-entry

qiankun 库的核心插件
在这里插入图片描述

import-html-entry 是 qiankun 实现 ​HTML Entry 模式的关键工具,具体功能包括:

  • ​请求并解析 HTML 文件
    从 HTML 中提取所有 <script><link> 标签的资源路径(包括外链和内联脚本)。
  • 资源分类与处理
    将资源分为 ​JS、CSS 和 ​入口脚本,并处理资源路径(如相对路径转绝对路径)。
  • 生成加载队列
    根据资源类型和依赖关系生成加载队列,确保资源按正确顺序加载。
  • 执行入口脚本并获取生命周期
    import-html-entry 通过劫持全局变量或模块系统,获取子应用导出的生命周期方法。
  • 执行环境隔离
    提供沙箱机制(如 Proxy),隔离子应用的 JS 执行环境。
    在这里插入图片描述

2. js沙箱

防止js全局污染
当在主应用环境中,开始加载子应用的 JS 文件(或内联脚本)时
qiankun 会通过 es6 Proxy 为该子应用创建一个 代理的window,

	// 创建一个虚拟的全局对象const fakeWindow = {};// 保存子应用对全局变量的修改this.modifiedProps = new Map();// 代理 window 对象
this.proxy = new Proxy(fakeWindow, {get: (target, prop) => {// 优先从子应用的修改中读取,否则从真实的 window 读取return this.modifiedProps.has(prop) ? this.modifiedProps.get(prop): window[prop];},set: (target, prop, value) => {// 所有对 window 的修改记录到 modifiedProps,不影响真实 windowthis.modifiedProps.set(prop, value);return true;},
  • ​读操作:子应用读取 window.xxx 时,优先从 modifiedProps 中获取子应用自己修改的值;若未修改,则从真实 window 读取。
  • ​写操作:子应用修改 window.xxx 时,值被记录到 modifiedProps,不会影响真实 window。

3. 样式隔离

  1. ​ 严格样式隔离(Shadow DOM):==​创建一个封闭的 DOM 树,外部样式无法影响内部,内部样式也不会泄漏到外部。
    • 配置方式:
    start({sandbox: {strictStyleIsolation: true, // 启用 Shadow DOM},
    })
    
    什么是Shadow DOM?是 Web Components 技术中的一部分
    传统 DOM 是全局可见的,而 Shadow DOM 允许在组件内部创建一个私有的 DOM 子树,独立于主文档树。外部样式和脚本无法直接访问 Shadow DOM 内部的内容。

通过 element.attachShadow({ mode }) 方法创建:

const host = document.createElement('div');// mode 可选 'open' 或 'closed'
// 是否允许外部通过 host.shadowRoot 访问 Shadow DOM。
const shadow = host.attachShadow({ mode: 'open' }); 
shadow.innerHTML = `<style>button { background: black; }</style><button>Shadow DOM 按钮</button>
`;
// 添加到页面后,按钮的样式不会受外部 CSS 影响
  • 优点:完全隔离,最严格的样式保护。
  • ​缺点:
    兼容性问题(部分旧浏览器不支持)。
    子应用需适配 Shadow DOM 环境(如事件冒泡路径变化)
  1. 动态样式隔离(Scoped CSS)
    为子应用的 CSS 选择器添加唯一前缀,限制其作用域。

    • 配置方式:
    start({sandbox: {experimentalStyleIsolation: true, // 启用动态作用域},
    })
    

    原理:
    子应用的根容器会被自动添加唯一属性(如 data-qiankun=“childApp”):

    <div data-qiankun="childApp"><!-- 子应用 DOM --><button class="button">Submit</button>
    </div>
    

    子应用的所有 CSS选择器(如 .button)添加上具有唯一属性:

    .button { background: red; }/* 重写后 */
    [data-qiankun="childApp"] .button { background: red; }
    
  • 优点:无需破坏子应用原有逻辑,兼容性好。
  • ​缺点:
    无法处理全局样式(如 body 或 html 的样式)。
    CSS-in-JS 或动态生成的样式可能需要额外处理。

如果子应用使用了 CSS-in-JS、内联样式或第三方 UI 库,可以结合以下方式彻底避免冲突:

  1. 子应用使用 CSS Modules 自动生成唯一的类名:
// 子应用组件
import styles from './App.module.css';function App() {return <div className={styles.title}>Child App</div>;
}
  1. ​CSS-in-JS
    通过 Styled-components 或 Emotion 生成作用域样式:
const Title = styled.div`color: red;
`;
// 生成唯一的 class 名称(如 .css-1x2y3)
  1. ​UI 库前缀
    修改第三方 UI 库的类名前缀(如 Ant Design):
    // 子应用中修改 less 变量
    @ant-prefix: 'custom-prefix';
    

4. 路由(基于single-spa)

single-spa 对原生浏览器路由 做了以下事情;

  1. 监听的路由事件: hashchange 和 popstate

    hashchange 和 popstate 是唯一原生支持无刷新路由的事件​​。
    单页应用(SPA)和微前端架构中的路由切换 依靠这两个api

    具体来说,监听这两个事件有以下几个关键目的:

    • 拦截路由变化 :
      当用户在浏览器中导航(如点击前进/后退按钮、修改 URL hash、或通过 History API 进行导航)时,single-spa 需要知道这些变化,以便决定哪些微前端应用需要被加载、挂载或卸载。
    • 控制事件触发顺序 :
      通过捕获这些事件的监听器,single-spa 可以确保在完成应用的挂载/卸载操作后,才将这些事件传递给各个微前端应用,避免应用在不适当的时机接收到路由事件。
    • 协调多个应用 :
      当路由变化时,可能需要卸载一些应用并挂载其他应用。通过监听这些事件,single-spa 可以协调这个过程,确保应用之间的平滑切换。
    • 统一的路由处理机制 :
      无论是 hash 路由还是 history 路由,single-spa 都能通过监听这两个事件来处理,为不同类型的路由提供统一的处理机制。
  2. 修补 History API
    具体来说,修补 History API 有以下几个关键目的:

    • 修补 pushState 或 replaceState 方法
      这两个方法修改了URL,默认不触发 popstate,通过修补这些方法,single-spa 可以在它们被调用后手动触发一个人工的 popstate 事件,确保所有应用都能感知到路由变化。

总的来说

  • 完善原生 URL的路由响应机制
  • 协调 子应用在不同URL间的挂载卸载

5. publicPath

静态文件在浏览器中​​被请求时的基础 URL 路径​​。
eg:

module.exports = {output: {filename: 'bundle.js',path: path.resolve(__dirname, 'dist'), // 与服务器无关,仅决定本地输出位置。publicPath: '/assets/', // 浏览器访问路径 需在前面加上 /assets},
};

若部署在服务器根目录,访问资源时会请求:http://xxx.com/assets/bundle.js。


在微应用中:
当子应用被主应用加载时,其资源路径可能相对于主应用的 URL,而非子应用的真实部署路径。

  • 示例​​:子应用部署在 http://child.com/sub-app/,但主应用在 http://main.com/ 加载子应用,此时子应用的 JS/CSS 可能错误地从 http://main.com/js/app.js 加载(404 错误)。

通过配置 publicPath,子应用可以区分当前是独立运行还是被主应用加载,动态修正资源路径。


⚠️ 且需要与子应用路由配置的协同:

问题场景​​:
如果子应用不配置路由的 basename(基础路径),当主应用通过 /sub-app 加载子应用时,
子应用的路由会直接拼接在 请求地址的根路径(/)下,导致路径匹配丢失。例如:

  • 主应用匹配子应用路径:http://main.com/sub-app
  • 子应用内部路由:/user/list
  • 访问子应用实际路径​​:http://main.com/user/list(错误,跳出了主应用的上下文)
  • ​​预期路径​​:http://main.com/sub-app/user/list(正确)

总结:为什么必须协同?​​
​​静态资源路径​​(publicPath)和 ​​路由路径​​(basename)共同决定了子应用的完整上下文:

  • publicPath 解决​​资源从哪里加载​​。
  • basename 解决​​路由在哪里生效​​。

两者缺一不可,否则会导致:

  • 资源加载成功,但页面渲染错位。
  • 路由跳转正确,但资源加载失败。

相关文章:

微前端随笔

✨ single-spa&#xff1a; js-entry 通过es-module 或 umd 动态插入 js 脚本 &#xff0c;在主应用中发送请求&#xff0c;来获取子应用的包&#xff0c; 该子应用的包 singleSpa.registerApplication({name: app1,app: () > import(http://localhost:8080/app1.js),active…...

【36期获取股票数据API接口】如何用Python、Java等五种主流语言实例演示获取股票行情api接口之沪深A股当天逐笔大单交易数据及接口API说明文档

​ 在量化分析领域&#xff0c;实时且准确的数据接口是成功的基石。经过多次实际测试&#xff0c;我将已确认可用的数据接口分享给正在从事量化分析的朋友们&#xff0c;希望能够对你们的研究和工作有所帮助&#xff0c;接下来我会用Python、JavaScript&#xff08;Node.js&…...

C++中的浅拷贝和深拷贝

浅拷贝只是将变量的值赋予给另外一个变量&#xff0c;在遇到指针类型时&#xff0c;浅拷贝只会把当前指针的值&#xff0c;也就是该指针指向的地址赋予给另外一个指针&#xff0c;二者指向相同的地址&#xff1b; 深拷贝在遇到指针类型时&#xff0c;会先将当前指针指向地址包…...

二叉树与红黑树核心知识点及面试重点

二叉树与红黑树核心知识点及面试重点 一、二叉树 (Binary Tree) 1. 基础概念 定义&#xff1a;每个节点最多有两个子节点&#xff08;左子节点和右子节点&#xff09; 术语&#xff1a; 根节点&#xff1a;最顶层的节点 叶子节点&#xff1a;没有子节点的节点 深度&#xf…...

GitHub 趋势日报 (2025年04月01日)

GitHub 趋势日报 (2025年04月01日) 本日报由 TrendForge 系统生成 https://trendforge.devlive.org/ &#x1f4c8; 今日整体趋势 Top 10 排名项目名称项目描述今日获星语言1punkpeye/awesome-mcp-serversA collection of MCP servers.⭐ 3280未指定2th-ch/youtube-musicYouTu…...

Java的SeleniumChromeDriver的常用方法

启动和关闭浏览器&#xff1a; driver.get(url)&#xff1a;打开指定的URL。driver.quit()&#xff1a;关闭浏览器并结束ChromeDriver会话。 元素定位&#xff1a; driver.findElement(By.id("elementId"))&#xff1a;通过元素的ID定位。driver.findElement(By.cl…...

字符串、列表、元组、字典

字符串 双引号或者单引号中的数据&#xff0c;就是字符串 字符串输入 之前在学习input的时候&#xff0c;通过它能够完成从键盘获取数据&#xff0c;然后保存到指定的变量中&#xff1b; 注意&#xff1a;input获取的数据&#xff0c;都以字符串的方式进行保存&#xff0c;即…...

【GEE学习笔记】报错解决:“Image.select: Band pattern ‘QA60‘ did not match any bands”

【GEE学习笔记】报错解决&#xff1a;“Image.select: Band pattern ‘QA60’ did not match any bands” 【GEE学习笔记】报错解决&#xff1a;“Image.select: Band pattern ‘QA60’ did not match any bands” 文章目录 【GEE学习笔记】报错解决&#xff1a;“Image.selec…...

AI可以赋能的三农产品、机械与服务

三农赛道涵盖农业、农村和农民相关的产品与服务&#xff0c;涉及农资、农业机械、智能设备、农产品加工及数字化服务等多个领域。随着人工智能&#xff08;AI&#xff09;技术的飞速发展&#xff0c;AI正在通过赋能农业的生产、管理、销售等各个环节&#xff0c;推动传统农业向…...

ngx_timezone_update

定义在 src\os\unix\ngx_time.c void ngx_timezone_update(void) { #if (NGX_FREEBSD)if (getenv("TZ")) {return;}putenv("TZUTC");tzset();unsetenv("TZ");tzset();#elif (NGX_LINUX)time_t s;struct tm *t;char buf[4];s tim…...

车载诊断架构 --- 整车重启先后顺序带来的思考

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 周末洗了一个澡,换了一身衣服,出了门却不知道去哪儿,不知道去找谁,漫无目的走着,大概这就是成年人最深的孤独吧! 旧人不知我近况,新人不知我过…...

GESP C++三级 知识点讲解

C编程三级标准 (一)知识点详述 (1)了解二进制数据编码:原码、反码、补码。 (2)掌握数据的进制转换:二进制、八进制、十进制、十六进制。 (3)掌握位运算:与(&)、或(|)、非(~)、异或(^)、左移(<<)、右移(>>)的基本使用方法及原理。 (4)了解算法的概念与描述&…...

前端 vs 后端:技术分工详解——从用户界面到系统逻辑的全解析

前端&#xff08;Frontend&#xff09; 和 后端&#xff08;Backend&#xff09; 是软件开发中两个核心概念&#xff0c;分别对应用户直接交互的部分和系统背后的逻辑处理部分。它们共同构成完整的应用程序&#xff0c;但分工不同。 目录 一、前端&#xff08;Frontend&#xf…...

Redis 除了数据类型外的核心功能 的详细说明,包含事务、流水线、发布/订阅、Lua 脚本的完整代码示例和表格总结

以下是 Redis 除了数据类型外的核心功能 的详细说明&#xff0c;包含事务、流水线、发布/订阅、Lua 脚本的完整代码示例和表格总结&#xff1a; 1. Redis 事务&#xff08;Transactions&#xff09; 功能描述 事务通过 MULTI 和 EXEC 命令将一组命令打包执行&#xff0c;保证…...

JavaScript智能对话机器人——企业知识库自动化

引言 内部知识管理常面临信息分散、查找困难的问题。本文将使用Node.js和虎跃办公的智能对话API&#xff0c;构建企业级知识问答机器人&#xff0c;支持自然语言查询和自动学习。 核心技术 自然语言处理&#xff08;NLP&#xff09;意图识别机器学习模型微调REST API集成 代…...

JS实现AES和DES

目录 目标 概述 DES AES 实战 JS实现DES JS实现AES 目标 了解AES和DES的特点并用JS实现。 概述 DES 翻译过来叫数据加密标准。它有5种加密模式&#xff08;CTR、OFB、CFB、CBC、ECB&#xff09;&#xff0c;在JS中&#xff0c;不同加密模式语法结构几乎一致&#xff0c…...

【C++11(下)】—— 我与C++的不解之缘(三十二)

前言 随着 C11 的引入&#xff0c;现代 C 语言在语法层面上变得更加灵活、简洁。其中最受欢迎的新特性之一就是 lambda 表达式&#xff08;Lambda Expression&#xff09;&#xff0c;它让我们可以在函数内部直接定义匿名函数。配合 std::function 包装器 使用&#xff0c;可以…...

Windows 10/11系统优化工具

家庭或工作电脑使用时间久了&#xff0c;会出现各种各样问题&#xff0c;今天给大家推荐一款专为Windows 10/11系统设计的全能优化工具&#xff0c;该软件集成了超过40项专业级实用程序&#xff0c;可针对系统性能进行深度优化、精准调校、全面清理、加速响应及故障修复。通过系…...

浅谈在HTTP中GET与POST的区别

从 HTTP 报文来看&#xff1a; GET请求方式将请求信息放在 URL 后面&#xff0c;请求信息和 URL 之间以 &#xff1f;隔开&#xff0c;请求信息的格式为键值对&#xff0c;这种请求方式将请求信息直接暴露在 URL 中&#xff0c;安全性比较低。另外从报文结构上来看&#xff0c…...

LightRAG实战:轻松构建知识图谱,破解传统RAG多跳推理难题

作者&#xff1a;后端小肥肠 &#x1f34a; 有疑问可私信或评论区联系我。 &#x1f951; 创作不易未经允许严禁转载。 姊妹篇&#xff1a; 2025防失业预警&#xff1a;不会用DeepSeek-RAG建知识库的人正在被淘汰_deepseek-embedding-CSDN博客 从PDF到精准答案&#xff1a;Coze…...

C++多线程编码二

1.lock和try_lock lock是一个函数模板&#xff0c;可以支持多个锁对象同时锁定同一个&#xff0c;如果其中一个锁对象没有锁住&#xff0c;lock函数会把已经锁定的对象解锁并进入阻塞&#xff0c;直到多个锁锁定一个对象。 try_lock也是一个函数模板&#xff0c;尝试对多个锁…...

垃圾回收——三色标记法(golang使用)

三色标记法(tricolor mark-and-sweep algorithm)是传统 Mark-Sweep 的一个改进&#xff0c;它是一个并发的 GC 算法&#xff0c;在Golang中被用作垃圾回收的算法&#xff0c;但是也会有一个缺陷&#xff0c;可能程序中的垃圾产生的速度会大于垃圾收集的速度&#xff0c;这样会导…...

Linux学习笔记——零基础详解:什么是Bootloader?U-Boot启动流程全解析!

零基础详解&#xff1a;什么是Bootloader&#xff1f;U-Boot启动流程全解析&#xff01; 一、什么是Bootloader&#xff1f;&#x1f4cc; 举个例子&#xff1a; 二、U-Boot 是什么&#xff1f;三、U-Boot启动过程&#xff1a;分为两个阶段&#x1f539; 第一阶段&#xff08;汇…...

Windows环境下开发pyspark程序

Windows环境下开发pyspark程序 一、环境准备 1.1. Anaconda/Miniconda&#xff08;Python环境&#xff09; 如果不怕包的版本管理混乱&#xff0c;可以直接使用已有的Python环境。 需要安装anaconda/miniconda&#xff08;python3.8版本以上&#xff09;&#xff1a;Anaconda…...

thinkphp8.0上传图片到阿里云对象存储(oss)

1、开通oss,并获取accessKeyId、accessKeySecret <!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><tit…...

SSM婚纱摄影网的设计

&#x1f345;点赞收藏关注 → 添加文档最下方联系方式咨询本源代码、数据库&#x1f345; 本人在Java毕业设计领域有多年的经验&#xff0c;陆续会更新更多优质的Java实战项目希望你能有所收获&#xff0c;少走一些弯路。&#x1f345;关注我不迷路&#x1f345; 项目视频 SS…...

1110+款专业网站应用程序UI界面设计矢量图标figma格式素材 Icon System | 1,100+ Icons Easily Customize

1110款专业网站应用程序UI界面设计矢量图标figma格式素材 Icon System | 1,100 Icons Easily Customize 产品特点 — 24 x 24 px 网格大小 — 2px 线条描边 — 所有形状都是基于矢量的 — 平滑和圆角 — 易于更改颜色 类别 &#x1f6a8; 警报和反馈 ⬆️ 箭头 &…...

nacos的地址应该配置在项目的哪个文件中

在 Spring Boot 和 Spring Cloud 的上下文中&#xff0c;​Nacos 的地址既可以配置在 bootstrap.yml 中&#xff0c;也可以配置在 application.yml 中&#xff0c;但具体取决于使用场景和需求。以下是两者的区别和最佳实践&#xff1a; ​1. bootstrap.yml vs application.yml …...

Llama 4 家族:原生多模态 AI 创新的新时代开启

0 要点总结 Meta发布 Llama 4 系列的首批模型&#xff0c;帮用户打造更个性化多模态体验Llama 4 Scout 是有 170 亿激活参数、16 个专家模块的模型&#xff0c;同类中全球最强多模态模型&#xff0c;性能超越以往所有 Llama 系列模型&#xff0c;能在一张 NVIDIA H100 GPU 上运…...

OpenCV 在树莓派上进行实时人脸检测

这段 Python 代码借助 OpenCV 库实现了在树莓派上进行实时人脸检测的功能。它会开启摄像头捕获视频帧&#xff0c;在每一帧里检测人脸并以矩形框标记出来&#xff0c;同时在画面上显示帧率&#xff08;FPS&#xff09;。 依赖库 cv2&#xff1a;OpenCV 库&#xff0c;用于计算…...