• 微前端-monorepo-无界



    前言

    本文主要记录微前端框架 无界 的使用与理解以及monorepo代码管理方式。


    一、微前端

    如今项目体积越来越大,导致维护成本增加,后端以微服务形式将项目拆成不同的服务,前端也需要对项目进行拆解,以达到长久发展的可能性。
    在我理解中,微前端就是将前端以某种规则进行隔离,在隔离中也能进行远程访问其他小应用,不受技术栈的影响,带来更多的可能性与可维护性。


    二 、monorepo

    • monorepo是一个单仓库多应用的代码管理方式,多个项目集成到一个仓库下,共享工程配置,同时又快捷地共享模块代码。
    • 中大型项目,多模块项目monorepo更适合管理代码,对调试、开发都有很大的帮助。

    注:在使用monorepo方式管理项目时,会产生诸多问题,如幽灵依赖、依赖安装耗时的问题,需要使用pnpm进行解决这类问题。


    三 、pnpm

    • 项目依赖在基于npm时,每个项目都有自己的项目依赖,都需要保存在本地,而pnpm则时将相同依赖项目的依赖包放在同一个位置,让不同项目共享同一个个依赖包,节约硬盘空间,提升了安装速度。
    • 针对不同版本的依赖,则只会保存不同的文件,不会将所有文件都保存

    硬链接

    共享了同一个硬盘地址
    pnpm通过硬链接直接指向 pnpm仓库 依赖包在硬盘的地址

    软链接(符号链接)

    只会记录路径,不占用资源
    如果依赖包之间有相互依赖,则会通过软连接指向该依赖路径,不会指向硬盘地址。

    pnpm

    幽灵依赖

    npm/yarn 安装依赖时,存在依赖提升,某个项目使用的依赖,并没有在其 package.json 中声明,也可以直接使用,这种现象称之为 “幽灵依赖”;随着项目迭代,这个依赖不再被其他项目使用,不再被安装,使用幽灵依赖的项目,会因为无法找到依赖而报错。

    使用pnpm可以解决此类问题

    依赖安装耗时长

    MonoRepo 中每个项目都有自己的 package.json 依赖列表,随着 MonoRepo 中依赖总数的增长,每次 install 时,耗时会较长。

    MonoRepo 中每个项目都有自己的 package.json 依赖列表,随着 MonoRepo 中依赖总数的增长,每次 install 时,耗时会较长。

    monorepo项目搭建

    项目构建需要一个注应用来搭载各个子应用,在安装依赖时需要在最外层创建 pnpm-workspace.yaml 配置文件,通过配置就可以轻松的下载所有项目的依赖。这样会将子项目中公共的依赖提取到外层modules中
    项目结构

    packages:
      # all packages in direct subdirs of packages/
      - 'main'
      # all packages in subdirs of components/
      - 'web/**'
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 在外层目录下可以通过 pnpm -F <子项目文件名> <命令> 的形式执行子项目中的命令

    子模块复用

    • 在外层创建common文件,将应用公共的模块提取到该文件夹下,比如axios等。
    • 在主项目main目录下,执行 pnpm -F <项目名称> add common

    commob依赖

    这样就可以直接在项目中使用common中的东西

    import {a} from 'common'
    
    • 1

    四、无界

    对于一些常见的微前端框架(qiankun、micro-app、EMP),对vite都不太友好,所以这里采用无界来搭建微前端框架

    特点

    • 接入简单只需要四五行代码
    • 不需要针对vite额外处理
    • 预加载
    • 应用保活机制

    不足

    • 隔离js使用一个空的iframe进行隔离
    • 子应用axios需要自行适配
    • iframe沙箱的src设置了主应用的host,初始化iframe的时候需要等待iframe的location.orign从’about:blank’初始化为主应用的host,这个采- - 用的计时器去等待的不是很优雅。
    • 底层原理 使用shadowDom 隔离css,js使用空的iframe隔离,通讯使用的是proxy

    接入无界

    • 在主项目main目录下安装wujie
    pnpm i wujie
    
    • 1
    • 在主项目main目录下安装对应类型项目的封装插件,该插件是对wujie的封装,从而使wujie的使用更快捷
    pnpm i wujie-vue3
    
    • 1
    • 在主应用main.ts中引入对应的wujie框架,并注册
    import Wujie from 'wujie-vue3'
    app.use(router).use(Wujie)
    
    • 1
    • 2
    • 子应用的使用,在主应用中使用WujieVue标签进行使用,在url中传入应用启动地址
      <WujieVue url="http://localhost:5174/" name="vue3"></WujieVue> 
      <WujieVue url="http://localhost:5175/" name="react"></WujieVue> 
    
    • 1
    • 2

    界面效果

    注:官网只对vue和react进行了封装其他框架需要自行封装

    无界预加载

    预加载能力可以极大的提升子应用打开的首屏时间

    只需要在preloadApp中将 exec 设置为true即可

    在主项目下main.ts文件中配置exec

    preloadApp({url:"http://localhost:5174/", name:"vue3",exec: true})
    preloadApp({url:"http://localhost:5175/", name:"react",exec: true})
    
    • 1
    • 2

    由于子应用提前渲染可能会导致阻塞主应用的线程,所以无界提供了类似 react-fiber 方式来防止阻塞线程
    react-fiber模拟了requestIdleCallback API ,预加载都被requestIdleCallback包裹,在浏览器空闲的时候提前预加载一些内容
    requestIdleCallback API会有两种情况:

    • 浏览器在每一帧执行一系列任务,如果有剩余时间就执行 requestIdleCallback
      • 处理用户事件
      • 执行定时器任务
      • 执行requestAnimationFrame动画
      • 执行dom的回流与重绘
      • 计算更新图层的绘制指令
      • 绘制指令合并主线程
    • 浏览器没有任务执行,会有50ms空余时间去执行 requestIdleCallback

    无界传参

    无界的通讯系统又三种方式进行通讯分别是:

    • props通讯

    主应用可以通过props注入数据和方法:

    <WujieVue name="xxx" url="xxx" :props="{ data: xxx, methods: xxx }"></WujieVue>
    
    • 1

    子应用可以通过$wujie来获取:

    const props = window.$wujie?.props; // {data: xxx, methods: xxx}
    
    • 1

    props通讯

    • window通讯
      主应用调用子应用的全局数据
    window.document.querySelector("iframe[name=子应用id]").contentWindow.xxx;
    
    • 1

    子应用调用主应用的全局数据

    window.parent.xxx;
    
    • 1

    window通讯

    • eventBus通讯
      发布订阅模式的通讯
      主应用使用方式:
    // 如果使用wujie
    import { bus } from "wujie";
    
    // 如果使用wujie-vue
    import WujieVue from "wujie-vue";
    const { bus } = WujieVue;
    
    // 如果使用wujie-react
    import WujieReact from "wujie-react";
    const { bus } = WujieReact;
    
    // 主应用监听事件
    bus.$on("事件名字", function (arg1, arg2, ...) {});
    // 主应用发送事件
    bus.$emit("事件名字", arg1, arg2, ...);
    // 主应用取消事件监听
    bus.$off("事件名字", function (arg1, arg2, ...) {});
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18

    子应用使用方式:

    // 子应用监听事件
    window.$wujie?.bus.$on("事件名字", function (arg1, arg2, ...) {});
    // 子应用发送事件
    window.$wujie?.bus.$emit("事件名字", arg1, arg2, ...);
    // 子应用取消事件监听
    window.$wujie?.bus.$off("事件名字", function (arg1, arg2, ...) {});
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    总结

    本文主要记录了微前端无界的使用与理解。

  • 相关阅读:
    汽车线束行业调研:预计2028年将达到615亿美元
    查询或解析solidity智能合约事件event或logs日志
    字符串漏洞注入深入学习
    数字时代 低代码赋能新零售系统
    【剑指 Offer II 032. 有效的变位词】
    Pytorch 实战 LESSON 4 张量的线性代数运算
    【Vue3教程】创建你的第一个Vue 3项目
    从0开始学汇编第一天:基础知识
    【算法详解】如何使用递归,递归使用的技巧详解
    我的 2023 年,35岁、父亲肺癌,失业,失恋、上岸
  • 原文地址:https://blog.csdn.net/smznbhh/article/details/132575625