一篇文章,让你全面认识微前端!谁说简历没东西写的?

大厂技术  高级前端  Node进阶

点击上方 程序员成长指北,关注公众号
回复1,加入高级Node交流群

对微前端的理解

微前端的核心理念是将前端应用程序看作是一个整体,由多个独立的部分组成。每个部分被视为一个微前端应用,它们可以具有自己的技术栈、开发流程和团队组织。这种方式使得团队可以独立开发和部署各个子应用,减少了协调和合并的复杂性。

为什么 Iframe 无法胜任微前端的工作?

IFrame 在传统的前端开发中是一种常见的技术,用于在页面中嵌入其他网页或应用程序。然而,在微前端架构中,IFrame 并不是一个理想的选择,主要是因为以下几个方面的限制:

  1. 隔离性和通信复杂性:IFrame 本身提供了一种隔离的环境,但这也带来了通信和数据交互的复杂性。由于每个子应用都在独立的 IFrames 中运行,它们之间的通信需要通过特定的机制,如消息传递,而这增加了开发和维护的复杂性。

  2. 性能和加载时间:每个 IFrames 都需要加载和渲染独立的 HTML、CSS 和 JavaScript。这意味着在加载微前端应用时,需要同时加载多个 IFrames,导致额外的网络请求和页面资源占用,可能会影响性能和加载时间。

  3. 样式和布局限制:IFrame 的内容在页面中是独立的,它们具有自己的 CSS 样式和布局上下文。这导致在微前端架构中难以实现全局样式的一致性,以及子应用之间的布局和交互的协调问题。

  4. 浏览器安全性限制:由于安全策略的限制,IFrame 之间的跨域通信可能受到限制,特别是在涉及跨域资源访问和共享数据时。这可能导致在微前端架构中需要处理复杂的安全性问题。

鉴于以上限制,微前端架构通常采用其他技术手段来实现子应用的拆分和集成,例如使用 Web Components、JavaScript 模块加载器等。这些技术能够提供更好的隔离性、通信机制和性能优化,使得微前端架构更具可行性和灵活性。

微前端运行原理

  • 监听路由变化

  • 匹配子应用

  • 加载子应用

  • 渲染子应用

监听路由变化

  1. 监听 hash 路由: window.onhashchange

  2. 监听 history 路由

history.go、history.back、history.forward 使用 popstate 事件 window.onpopstate

监听的方式

window.addEventListener('popstate', () => {})

重写: pushState、replaceState 需要通过函数重写的方式进行 劫持

const rawPushState = window.history.pushState
window.history.pushState = function(...args) {
  rawPushState.apply(window.history, args)

  // 其他逻辑
}

const rawReplaceState = window.history.replaceState
window.history.replaceState = function(...args) {
  rawReplaceState.apply(window.history, args)

  // 其他逻辑
}

在 Vue 项目中,我们通过 this.router.push会触发‘history.pushState‘事件,this.router.push 会触发 history.pushState 事件,this.router.push会触发‘history.pushState‘事件,this.router.replace 会触发 history.replaceState 事件。

匹配子应用

监听路由的变化后,拿到当前路由的路径 window.location.pathname,然后根据 registerMicroApps 的参数 apps 查找子应用。因为子应用都配置了 activeRule

// 如果当前的 pathname 以 activeRule 开头,表明匹配到了子应用

const currentApp = apps.find(app => window.location.pathname.startWith(app.activeRule))

加载子应用

当我们找到了与当前路由匹配的子应用,接着就去加载这个子应用的资源。

function handleRouter = async () => {
  // 匹配子应用

  // 加载资源
  const html = await fetch(currentApp.entry.then(res => res.text())

  // 将 html 渲染到指定的容器内
  const container = document.querySelector(currentApp.container)
}

这个时候,我们就拿到了子应用的 html 文本。

但是我们不能给直接通过 container.innerHTML = html 将文本放到容器内,这样是无法显示的。

注意 浏览器处于安全考虑,放到页面上的 html 如果包含了 js 脚本,它是不会去执行 js 的。我们需要手动处理 script 脚本。

importHTML 加载资源/处理脚本

我们来封装一个函数 importHTML,专门来处理 html 文本。(qiankun内部引用的 import-html-entry 就是做这个事的。)

我们可以把加载子应用资源的 fetch 请求放到 importHTML 函数中,它还有如下几个功能:

  • 将获取到的 html 文本,放到 template DOM节点中

  • 获取所有的 Script 脚本

  • 执行所有的 Script 脚本

export const importHTML = url => {
  const html = await fetch(currentApp.entry).then(res => res.text()

  const template = document.createElement('div')

  template.innerHTML = html

  const scripts = template.querySelectAll('script')

  const getExternalScripts = () => {
    console.log('解析所有脚本: ', scripts)
  }

  const execScripts = () => {}

  return {
    template, // html 文本
    getExternalScripts, // 获取 Script 脚本
    execScripts, // 执行 Sript 脚本
  }
}

script 脚本分为 内联 脚本和外链脚本,这里需要分开处理,拿到内联脚本后,获取内容可以通过 eval 直接处理。如果是含有 scr 的 script 脚本,还需要拿到 src 的值,通过 fetch 去加载脚本。

我们在 getExternalScripts 方法中来处理

const getExternalScripts = async () => {
  return Promise.all(Array.from(scripts).map(script => {
    // 获取 scr 属性
    const src = script.getAttribute('src')

    if (!src) {
      return Promise.resolve(script.innerHTML)
    } else {
      return fetch(src.startWith('http') ? src : `${url}${src}`).then(res => res.text())
    }
  }))
}

然后我们就可以通过 execScripts 方法去调用 getExternalScripts,拿到所有的脚本内容后,执行!

const execScripts = async () => {
  const scripts = await getExternalScripts() 

  scripts.forEach(code => {
    eval(code)
  })
}

qiankun

qiankun.umijs.org/zh[1]

我们看 qiankun 的依赖,可以发现 qiankun 是基于 single-spa 实现的,通过 import-html-entry 包处理 html / css

"dependencies": {
  "import-html-entry": "^1.14.0",
  "single-spa": "^5.9.2"
  // ...
},

微前端的子项目是怎么引入到主项目里的

registerMicroApps(
  [
    {
      name: 'react16',
      entry: '//localhost:7100',
      container: '#subapp-viewport',
      loader,
      activeRule: '/react16',
    },
    {
      name: 'react15',
      entry: '//localhost:7102',
      container: '#subapp-viewport',
      loader,
      activeRule: '/react15',
    },
  ]
)
  1. 容器指定路由匹配规则加载子应用,一旦路径匹配就会加载子应用资源

  2. 子应用打包输出格式为 umd,并且要允许跨域

// vue.config.js

module.exports = {
  devServer: {
    // ...
    headers: {
      'Access-Control-Allow-Origin': '*',
    },
  },
  // 自定义webpack配置
  configureWebpack: {
    output: {
      // 把子应用打包成库文件、格式是 umd
      library: `${name}-[name]`,
      libraryTarget: 'umd',
      jsonpFunction: `webpackJsonp_${name}`,
    },
  },
}

umd

umd全称是UniversalModuleDefinition,是一种通用模块定义格式,通常用于前端模块化开发中。

由于不同的模块化规范定义不同,为了让各种规范的模块可以通用,在不同的环境下都可以正常运行,就出现了umd这个通用格式。

特点

umd 格式是一种既可以在浏览器环境下使用,也可以在 node 环境下使用的格式。它将 CommonJS、AMD以及普通的全局定义模块三种模块模式进行了整合。

(function (global, factory) {
  // CommonJS
 typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) :
  // AMD
 typeof define === 'function' && define.amd ? define(['exports'], factory) :
  // Window
 (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global.qiankun = {}));
}(this, (function (exports) {
  // 应用代码
})));

为什么 qiankun 要求子应用打包为 umd 库格式呢?

主要是为了主应用能够拿到子应用在 入口文件 导出的 生命钩子函数,这也是主应用和子应用之间通信的关键。

  • bootstrap

  • mount

  • unmount

获取子应用资源 - import-html-entry

HTML Entry + Sandbox 是 qiankun 区别于 single-spa 的主要两个特性。

single-spa和qiankun最大的不同,大概就是qiankun实现了html entry,而single-spa只能是js entry

通过 import-html-entry,我就能像 iframe 一样加载一个子应用,只需要知道其 html 的 url 就能加载到主应用中。

importHTML 几个核心方法:

首先importHTML的参数为需要加载的页面url,拿到后会先通过 fetch方法 读取页面内容。

import importHTML from 'import-html-entry';

importHTML('./subApp/index.html')
  .then(res => {
    console.log(res.template);

    res.execScripts().then(exports => {
      const mobx = exports;
      const { observable } = mobx;
      observable({
        name: 'kuitos'
      })
    })
});

返回值

  • template - string - 处理过的 HTML 模板。

  • assetPublicPath - string - 资源的公共途径。

  • getExternalScripts - Promise<string[]> - 来自模板的脚本 URL。

  • getExternalStyleSheets - Promise<string[]> - 来自模板的 StyleSheets URL。

  • execScripts - (sandbox?: object, strictGlobal?: boolean, execScriptsHooks?: ExecScriptsHooks): - Promise - the return value is the last property on window or proxy window which set by the entry - script.

    • sandbox - optional, Window or proxy window.

    • strictGlobal - optional, Strictly enforce the sandbox.

processTpl

它会解析html的内容并且删除注释,获取style样式及script代码。通过大量的正则 + replace,每一个步骤都做了很多适配,比如获取script脚本,需要区分该script是不是entry script,type是JavaScript还是module,是行内script还是外链script,是相对路径还是绝对路径,是否需要处理协议等等。

processTpl的返回值有 template,script,style,entry。

JS 沙箱

JavaScript 沙箱是一种安全机制,用于隔离和限制 JavaScript 代码的执行环境,以防止恶意代码或意外行为对系统造成损害。沙箱提供了一种受控的环境,限制了代码的访问权限和执行能力,确保代码只能在受限制的范围内操作。

JavaScript 沙箱通常用于以下情况:

  1. 在多租户环境中,确保不同用户或组织的代码相互隔离,防止相互干扰或访问敏感信息。

  2. 在第三方代码或插件中使用,以确保其代码不会对宿主环境造成潜在的安全漏洞或冲突。

  3. 在应用程序中执行用户提供的代码,例如在线代码编辑器或脚本执行环境,以防止恶意代码对用户数据或系统进行攻击。

  4. 在浏览器中执行不受信任的代码,例如浏览器插件或扩展,以保护用户的隐私和安全。

JavaScript 沙箱通过限制代码的访问权限、提供隔离的执行环境、使用安全策略和沙箱沙盒技术等手段来实现。常见的 JavaScript 沙箱技术包括沙盒环境、Web Worker、iframe、JavaScript 虚拟机等。这些技术通过限制代码的执行权限、提供独立的运行环境、隔离全局上下文等方式来确保代码的安全执行。

快照沙箱-SnapshotSandbox

缺点:

  • 遍历 window 上所有属性,性能差

  • 同一时间只能激活一个微应用

  • 污染全局 window

优点:

可以支持不兼容Proxy的浏览器。

先了解 SnapshotSandbox 的功能

  1. 我们激活沙箱后,在 window 上修改的所有属性,都应该存起来,在下一次激活时,需要还原上次在 window 上修改的属性

  2. 失活后,应该将 window 还原成激活前的状态

我们来举个例子

// 激活前
window.city = 'Beijing'

// 激活
sanbox.active()

window.city = '上海'

// 失活
sanbox.inactive()
console.log(window.city) // 打印 'Beijing'

// 再激活
console.log(window.city) // 打印 '上海'

接下来,实现一个简易版的 SnapshotSandbox

  1. SnapshotSandbox 能够还原 window 和记录自己以前的状态,那么就需要两个对象来存储这些信息

1. windowSnapshot 用来存储沙箱激活前的 window

2. modifyPropsMap 用来存储沙箱激活期间,在 window 上修改过的属性
  1. 沙箱需要两个方法及作用

1. sanbox.active() // 激活沙箱

  - 保存 window 的快照

  - 再次激活时,将 window 还原到上次 active 的状态

2. sanbox.inactive() // 失活沙箱

  - 记录当前在 window 上修改了的 prop

  - 还原 window 到 active 之前的状态

我们先来实现沙箱内部细节:

class SnapshotSandbox {

  constructor() {
    this.windowSnapshot = {}

    this.modifyPropsMap = {}
  }

  active() {
    // 1. 保存 window 的快照
    for (let prop in window) {
      if (window.hasOwnProperty(prop)) {
        this.windowSnapshot[prop] = window[prop]
      }
    }

    // 2. 再次激活时,将 window 还原到上次 active 的状态,modifyPropsMap 存储了上次 active 时在 widow 上修改了哪些属性
    Object.keys(modifyPropsMap).forEach(prop => {
      window[prop] = this.modifyPropsMap[prop]
    })
  }

  inactive() {
    for(let prop in window) {
      if (window.hasOwnProperty(prop)) {
        // 两者不相同,表示修改了某个 prop 记录当前在 window 上修改了的 prop
        if (window[prop] !== this.windowSnapshot[prop]) {
          this.modifyPropsMap[prop] = window[prop]
        }

        // 还原 window
        window[prop] = this.windowSnapshot[prop]
      }
    }
  }
}

我们来验证一下,首先设置 window.city 一个初始值 beijing,然后初始化 沙箱,在第一次激活后,修改了 window.city 为 上海,那么应该在失活后,打印 beijing,再次激活时,window.city 是 上海

window.city = 'beijing'

const ss = new SnapshotSandbox()

console.log('window.city0 ', window.city)

ss.active() // 激活

window.city = '上海'

console.log('window.city1 ', window.city) // 上海

ss.inactive()

console.log('window.city2 ', window.city) // beijing

ss.active()

console.log('window.city3 ', window.city) // 上海

ss.inactive()
console.log('window.city4 ', window.city) // beijing

ss.active()
console.log('window.city5 ', window.city) // 上海

不支持多个应用同时运行,因为污染了全局 window

window.city = 'beijing'

const ss = new SnapshotSandbox()
ss.active() // 激活
window.city = '上海'

const ss1 = new SnapshotSandbox()
ss1.active() // 激活

window.city = '广州'

console.log(window.city) // 广州

Legacy沙箱-LegacySandbox(单例)

  • 不需要遍历 window 上的所有属性,性能比快照沙箱要好

  • 基于 proxy 实现,依然操作了 window,污染了全局,同一时间只能运行一个应用

  • 兼容性没有快照沙箱好

功能和 快照沙箱 一样,但内部实现是通过 proxy 实现的。

ProxySandbox 沙箱(多例)

  • 基于 proxy 代理对象,不需要遍历 window,性能要比快照沙箱好

  • 支持多个应用

  • 没有污染全局 window

  • 应用失活后,依然可以获取到激活时定义的属性。

主要实现在 constructor 中,创建一个 fakeWindow 对象,通过 Proxy 代理这个对象,全程没有改变 window

只是在获取属性值的时候,如果在代理对象上没有找到想要的属性,才回去 window 中查找。

class ProxySandbox {

  constructor() {
    // 沙箱是否是激活状态
    this.isRunning = false

    const fakeWindow = Object.create(null)

    const _this = this

    this.proxyWindow = new Proxy(fakeWindow, {
      set(target, prop, value) {
        // 只有激活状态下,才做处理
        if (_this.isRunning) {
          target[prop] = value
          return true
        }
      },
      get(target, prop, reciver) {
        // 如果fakeWindow里面有,就从fakeWindow里面取,否则,就从外部的window里面取
        return prop in target ? target[prop] : window[prop]
      }
    })
  }

  active() {
    this.isRunning = true
  }

  inactive() {
    this.isRunning = false
  }
}

window.city = '北京'

const p1 = new ProxySandbox()
const p2 = new ProxySandbox()

// 激活
p1.active()
p2.active()

p1.proxyWindow.city = '上海'
p2.proxyWindow.city = '杭州'

console.log(p1.proxyWindow.city) // '上海'
console.log(p2.proxyWindow.city) // '杭州'
console.log(window.city) // 北京

// 失活
p1.inactive()
p2.inactive()

console.log(p1.proxyWindow.city) // '上海'
console.log(p2.proxyWindow.city) // '杭州'
console.log(window.city) // '北京'

qiankun 的样式问题

如果不启动样式隔离,主应用、子应用所有的样式都是全局环境下,意味着,如果我在主应用里面设置了高权重的 css 样式,是会直接影响到子应用的。

// 主应用 main.css
h1 {
  color: red !important;
}

button {
  background-color: red !important;
}

主应用、子应用所有的 h1 和 button 都会应用以上颜色。

当然我们不能这样做,我们的应用间样式应该独立,不能互相影响。可以通过 BEM 解决,不过在大型项目下,约定是一件很不靠谱的事情,最好是在框架中解决此问题,一劳永逸。

qiankun 样式隔离方案

  • shadow dom(sanbox: strictStyleIsolation)

  • scoped css(sanbox: experimentalStyleIsolation)

在 start 方法中,配置 sanbox 属性,即可开启 css 隔离。

// sanbox: boolean | { strictStyleIsolation?: boolean, experimentalStyleIsolation?: boolean }

start({
  sanbox: true
})
  • strictStyleIsolation

strictStyleIsolation 模式下 qiankun 会为每个微应用的容器包裹上一个 shadow dom 节点,**所有的子应用都被 #shadow-root 所包裹**,从而确保微应用的样式不会对全局造成影响。

当我们开启了 strictStyleIsolation 模式后,主应用设置的高权重 css 确实没有影响子应用了。但是,但是,咱们去看看 Vue dialog 的样式(别看 React 的,因为React事件在 Shadow DOM 中根本不起作用 😂😂😂)

注意:

shadow dom 并不是一个无脑的解决方案,特别是在 React 中,事件的处理可能不那么奏效了😄!

React 官方关于 web component 的解释[2]

乍一看是不是没问题?

df8d3602dc9f8893ec43ca6d0bd42bc9.jpeg
image.png

我们摁下电脑的 ESC 键,会触发 是否取消弹窗 的二次确认,你再看看有没有问题?

dc30a4727bec3b51838c35b44d81c778.jpeg样式完全丢失了,这是为什么呢?因为二次确认的 Dialog 是挂在 body 下,而我们整个子应用都被 shadow dom 所包裹,内部的样式对外部的样式起不到任何作用,所以这个弹窗失去了漂亮的外衣了😭😭😭

不过,为啥弹窗要挂在 body 下?

这个是为了避免被父元素的样式影响,比如父元素设置了 display:none,那么这个弹窗也是无法展示的。

  • experimentalStyleIsolation

experimentalStyleIsolation 被设置为 true 时,qiankun 会改写子应用所添加的样式为所有样式规则增加一个特殊的选择器规则来限定其影响范围,因此改写后的代码会表达类似为如下结构:

// 假设应用名是 react16 中的样式是这样
.app-main {
  font-size: 14px;
}

// ===== 处理后 ======>

div[data-qiankun-react16] .app-main {
  font-size: 14px;
}

有点类似 Vue 中的 css scoped 作用,给每个子应用加了一个 ”唯一“ 的属性选择器。

这个时候,React 的事件处理没问题了(真好啊😁),我们来到页面上看看效果:

67e265bfefd0ccf678078474334bcbfa.jpeg事件是生效了,但是弹窗样式丢失了😭😭😭

这个弹窗是挂在 body 下,而加了 experimentalStyleIsolation 之后,所有的样式都加了 div[data-qiankun="react16"] 前缀,唯独 body 下的 dialog 没有加前缀,导致无法应用到正确的样式了。(Vue子应用 也有这样的问题!!)

还有就是,在主应用设置的高权重样式依然影响到了子应用。

Vue Scoped

在 Vue 的单文件组件中使用 <style scoped> 标签时,Vue 会自动将该样式应用于当前组件的元素,并在编译过程中为每个 CSS 规则添加一个唯一的属性选择器,以确保样式仅对当前组件有效。

h3 {
  background-color: pink;
  color: blue;
}

// ======= 使用 style scoped 后 ====>

h3[data-v-469af010] {
  background-color: pink;
  color: blue;
}

CSS Modules

要使用 CSS Modules,首先要改造 webpack.config.js,修改 css-loader 部分

// webpack.config.js
{
  test: /.(le|c)ss$/,
  use: ['style-loader', {
    loader: 'css-loader',
    options: {
      modules: true // 开启 css modules
    }
  }, 'less-loader'],
},

然后定义 模块 css index.module.css

.text-color {
  color: red;
}

.text-gb {
  background-color: pink;
}

使用的时候,导入 index.module.css

import styles from './index.module.css'

<div className={styles['text-bg']}>
  <span className={styles['text-color']}>CSS Modules</span>
</div>

使用 CSS Modules 之后,选择器名字上加了 hash。

既然 父元素 class name 用 css modules 加上了 hash,那么内部元素的 class name 该怎么用怎么用,不使用 模块CSS 中的样式也是 OK 的。

css modules 和 scoped css 差不多,都能实现组件级别样式隔离,能设置子组件和全局样式,只是实现方式不同,导致了使用起来也有差异。

那么 qiankun 的样式隔离还有必要做吗

老项目还是可以用的,比如 JQuery 这种,qiankun 的样式隔离能用。

如果要在 JQuery 这种项目中使用 CSS Module、CSS in JS,可能改造成本有点大了。

只是现在的应用,不管是 vue 还是 react 基本都开启了组件级别样式隔离,qiankun 自带的样式隔离问题太多了,不能用了。

CSS 沙箱解决方案

在前端开发中,为了实现 CSS 的沙箱化,即将某个 CSS 样式应用于特定的范围而不影响其他元素,可以采用以下几种实现方案:

  1. 命名约定(BEM) :通过给特定范围内的元素添加特定的类名或命名前缀,然后在 CSS 中通过类选择器或属性选择器来应用相应的样式。这种方式需要在开发过程中遵循命名约定,确保样式只作用于指定的元素,避免与其他元素产生冲突。

<div class="block">
  <div class="block__element"></div>
  <div class="block__element--modifier"></div>
</div>

<div class="header-section">
  <div class="header-section__logo"></div>
  <div class="header-section__menu"></div>
</div>
  1. CSS Modules:CSS Modules 是一种在构建过程中将 CSS 样式模块化的解决方案。它通过为每个模块生成唯一的类名,将样式限定在模块的范围内。在使用 CSS Modules 时,可以在 JavaScript 代码中导入样式文件,并通过类名来引用特定的样式,从而实现样式的沙箱化。

要使用 CSS Modules,需要在 css-loader 中开启 module:

// webpack.config.js
{
  test: /.(le|c)ss$/,
  use: ['style-loader', {
    loader: 'css-loader',
    options: {
      modules: true // 开启 css modules
    }
  }, 'less-loader'],
}

Button.module.css

.button {
  background-color: blue;
  color: white;
}

.button--disabled {
  opacity: 0.5;
}

App.js

import React from 'react';
import styles from './Button.module.css';

interface ButtonProps {
  disabled?: boolean;
  onClick: () => void;
}

const Button: React.FC<ButtonProps> = ({ disabled, onClick, children }) => {
  const buttonClasses = `${styles.button} ${disabled ? styles['button--disabled'] : ''}`;

  return (
    <button className={buttonClasses} onClick={onClick} disabled={disabled}>
      {children}
    </button>
  );
};

export default Button;
  1. CSS-in-JS:CSS-in-JS 是一种将 CSS 样式写在 JavaScript 代码中的方式,通过将样式与组件绑定在一起,实现了样式的局部化和沙箱化。常见的 CSS-in-JS 解决方案包括 styled-components、Emotion、CSS Modules with React 等。

import React from 'react';
import styled from 'styled-components';

const Button = styled.button`
  background-color: blue;
  color: white;
  padding: 10px 20px;
  border: none;
  border-radius: 4px;
  cursor: pointer;

  &:hover {
    background-color: darkblue;
  }

  &:disabled {
    opacity: 0.5;
    cursor: not-allowed;
  }
`;

const ExampleComponent = () => {
  return (
    <div>
      <Button onClick={() => console.log('Button clicked')}>Click me</Button>
      <Button disabled>Disabled Button</Button>
    </div>
  );
};

export default ExampleComponent;

在使用 CSS-in-JS 的方案中,样式是通过 JavaScript 运行时动态生成的,每个组件都具有自己独特的样式,并且不会与其他组件的样式发生冲突。这种方式提供了更好的样式隔离和组件化能力,并且使得样式与组件的代码更紧密集成在一起,提高了可维护性和可读性。

  1. Shadow DOM:Shadow DOM 是 Web 标准中的一个技术,可以创建一个隔离的 DOM 子树,其中的样式和脚本不会影响到外部的 DOM。通过在元素上应用 Shadow DOM,可以将样式限定在 Shadow DOM 内部,实现样式的沙箱化。Shadow DOM 主要用于 Web 组件开发,可以实现组件样式的封装和隔离。

这些方案各有特点,选择合适的方案取决于具体的需求和项目情况。命名约定是最简单的方式,适用于小型项目和简单的样式隔离。CSS Modules 和 CSS-in-JS 提供了更丰富的功能和工具支持,适用于中大型项目和复杂的样式需求。Shadow DOM 则主要应用于 Web 组件开发,提供了更强大的封装和隔离能力。

<!DOCTYPE html>
<html>
<head>
  <style>
    .outer {
      background-color: pink;
      padding: 20px;
    }
  </style>
</head>
<body>
  <div class="outer">
    <h2>Outer Component</h2>
    <div id="inner-root"></div>
  </div>

  <script>
    const outerElement = document.querySelector('.outer');
    const innerRoot = document.getElementById('inner-root');

    // 创建 Shadow DOM,open 表示可以通过页面内的 JavaScript 方法来获取 Shadow DOM
    const shadowRoot = innerRoot.attachShadow({ mode: 'open' });

    // 在 Shadow DOM 中创建样式
    const style = document.createElement('style');
    style.textContent = `
      .inner {
        background-color: aqua;
        padding: 10px;
      }
    `;
    shadowRoot.appendChild(style);

    // 在 Shadow DOM 中创建内容
    const innerDiv = document.createElement('div');
    innerDiv.className = 'inner';
    innerDiv.textContent = 'Inner Component';
    shadowRoot.appendChild(innerDiv);
  </script>
</body>
</html>

微前端框架

qiankun、wujie、micro-app 的区别主要还是实现容器(或者叫沙箱)上有区别

  • qiankun: function + proxy + with

  • micro-app: web components

  • wujie: web components 和 iframe。

微前端(qiankun)架构中,主应用和子应用有共同的组件,如何封装呢?

在微前端架构中,主应用和子应用可能会共享一些组件,为了实现组件的共享和封装,可以采用以下方法:

  1. 封装为独立的 npm 包:将共享的组件封装为独立的 npm 包,并发布到私有或公共的 npm 仓库中。主应用和子应用都可以通过 npm 安装该组件,并在需要的地方引入和使用。

  2. Git 仓库依赖:将共享组件放置在一个独立的 Git 仓库中,并通过 Git 仓库的依赖关系来引入组件。主应用和子应用可以通过 Git 仓库的 URL 或路径来引入共享组件。

  3. Git Submodule:如果主应用和子应用都在同一个 Git 仓库下,可以使用 Git Submodule 的方式来引入共享组件。将共享组件作为子模块添加到主应用和子应用的仓库中。

  4. 本地引用:如果主应用和子应用处于同一个代码仓库中,可以直接通过相对路径引入共享组件。将共享组件放置在一个独立的目录下,并通过相对路径引用。

无论选择哪种方式,关键是要保持共享组件的独立性和可维护性。确保共享组件的代码和样式与具体的主应用和子应用解耦,避免出现冲突和依赖混乱的情况。同时,建议对共享组件进行版本管理,以便在更新和维护时能够更好地控制和追踪变更。

在微前端架构中,可以通过合适的方式引入共享组件,使主应用和子应用可以共享和复用组件,提高开发效率和代码质量。

monorepo架构

Monorepo 模式也可以解决微前端中的公共依赖包和公共组件的问题。

Monorepo 模式是指将多个项目或应用放置在同一个代码仓库中管理的开发模式。

在 Monorepo 中,可以将公共依赖包和公共组件作为共享资源,放置在代码仓库的合适位置,供不同的项目或应用使用。这样可以避免不同项目之间重复安装和维护相同的依赖包,也能够统一管理和更新公共组件。

以下是 Monorepo 模式下解决微前端中公共依赖包和公共组件的方式:

  1. 公共依赖包管理:将公共的依赖包放置在代码仓库的根目录或指定目录下,通过工具如 Yarn 或 Lerna 管理依赖包的安装、更新和版本控制。不同的项目或应用可以通过引用共享的依赖包来解决依赖关系,避免重复安装和冲突。

  2. 公共组件封装:将公共的组件封装为独立的包或模块,放置在代码仓库的特定目录中,并通过工具如 NPM 发布和安装。不同的项目或应用可以通过引用共享的组件来实现组件的复用和共享,提高开发效率和代码一致性。

通过 Monorepo 模式,可以集中管理和维护公共依赖包和公共组件,减少重复工作和资源浪费。同时,还能够促进团队协作和代码共享,统一规范和风格,提高整体项目的质量和可维护性。

作者:Hello_AlexCc
链接:https://juejin.cn/post/7242623208841592869
来源:稀土掘金

结语

 
 

Node 社群

我组建了一个氛围特别好的 Node.js 社群,里面有很多 Node.js小伙伴,如果你对Node.js学习感兴趣的话(后续有计划也可以),我们可以一起进行Node.js相关的交流、学习、共建。下方加 考拉 好友回复「Node」即可。

20514ded8372bffe8b3228f286d2e9a3.png

“分享、点赞、在看” 支持一波
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 狂神Vue是由中国前端开发者尤雨溪开发的一款流行的JavaScript框架。Vue具有简洁的语法、高效的性能和灵活的组件化开发模式,因此在前端开发领域得到了广泛的应用和认可。 在《狂神Vue笔记》这篇文章中,作者整理了对于Vue框架的学习和实践经验,旨在帮助读者更好地理解和运用Vue。文章首先介绍了Vue框架的基本概念和特点,如Vue实例、生命周期、模板语法等。然后详细讲解了Vue的核心功能和常用的基础知识,例如数据绑定、计算属性、事件处理等。接着,文章还介绍了Vue的路由、状态管理和组件化开发等进阶知识,以及常见的一些Vue插件和工具的使用。 文章作风格简洁明了,重点突出,对于每个知识点都给予了清晰的解释和实际示例。同时,狂神还根据自己的实际经验提供了一些实战技巧和开发中常见的问题解决方法。这些经验分享不仅有助于初学者快速上手,也为有一定经验的开发者提供了一些新的思路和技巧。 总之,《狂神Vue笔记》是一篇值得阅读的文章,无论是对Vue框架感兴趣的初学者,还是对于Vue有一定了解的开发者,都可以从中获得一些有益的知识和经验。通过学习这些笔记,读者可以更加深入地了解Vue框架的使用和原理,提升自己的前端开发能力。 ### 回答2: 《狂神Vue笔记.md》是一份关于Vue框架的学习笔记,由狂神团队撰而成。这份笔记详细地介绍了Vue的基本概念、核心特性和使用方法。一共包含了20个章节,内容全面且系统。 笔记的第一章主要介绍了Vue的基本概念,包括Vue实例、生命周期、指令等。第二章到第五章则讲解了Vue的模板语法、计算属性、侦听器、样式绑定等。通过学习这些章节,读者可以对Vue的基本语法和使用方式有一个清晰的认识。 接下来的几章介绍了Vue的组件化开发,包括组件的定义、组件之间的通信、插槽等。这些章节详细地介绍了Vue组件的相关概念和使用方法,使读者能够灵活地进行组件化开发。 笔记的后半部分则围绕Vue的高级特性展开,如路由、状态管理、动画等。这些章节深入探讨了Vue的高级用法和扩展性,对于希望进一步深入学习的读者来非常有帮助。 总的来,《狂神Vue笔记.md》是一份非常全面、详细的Vue学习资料。通过学习这份笔记,读者可以系统地掌握Vue框架的基本概念和核心特性,同时也能够了解到一些高级用法和扩展性。这份笔记适合初学者入门,也适合有一定经验的开发者进阶学习。读者可以通过实践和不断深入学习,更好地掌握和应用Vue框架。 ### 回答3: 《狂神Vue笔记》是一本非常有价值的学习资料。该书以清晰、简明的语言介绍了Vue.js框架的核心概念和使用方法,适合任何想要深入学习Vue.js的开发者。 首先,该书从Vue.js的基本概念开始讲解,包括Vue实例、模板语法和组件等。通过实例代码和明,读者可以直观地了解Vue.js的基本用法和原理。 其次,该书详细介绍了Vue.js的高级特性,例如Vue组件的通信方式、Vue路由和状态管理等。这些特性是Vue.js框架优势的体现,通过学习这些内容,读者可以更好地运用Vue.js开发复杂的应用程序。 此外,该书还涵盖了Vue.js框架的生态系统,介绍了Vue.js周边的工具和库,例如Vue CLI和Vue Router等。这些工具和库可以帮助开发者更高效地进行Vue.js项目开发,提高开发效率。 总之,通过《狂神Vue笔记》,读者可以系统地学习和掌握Vue.js框架的核心概念和使用方法。该书内容丰富,重点明确,适合初学者和有一定经验的开发者阅读。无论是想要进一步学习Vue.js还是应用Vue.js进行项目开发,这本书都是不可或缺的参考资料。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值