一文帶你進入微前端世界

什麼是微前端

微前端(Micro-Frontends)是一種類似於微服務的架構,它將微服務的理念應用於瀏覽器端,即將 Web 應用由單一的單體應用轉變爲多個小型前端應用聚合爲一的應用。微前端(micro-frontends)術語在 2016 年在 TECHNOLOGY RADAR[1] 中被提及。

微前端架構具備以下的特點:

微前端解決了什麼問題?

微前端架構旨在解決單體應用在一個相對長的時間跨度下,由於參與的人員、團隊的增多、變遷,從一個普通應用演變成一個巨石應用 (Frontend Monolith) 後,隨之而來的應用不可維護的問題。比如:

通過微前端拆分成一個容器應用和多個子應用之後,各個應用能夠獨立部署,相互之間隔離,從而做到:

微前端實現方案對比

Nginx 路由轉發

通過 Nginx 配置反向代理來實現不同路徑映射到不同應用,例如 www.abc.com/app1 對應 app1,www.abc.com/app2 對應 app2,這種方案本身並不屬於前端層面的改造,更多的是運維的配置

優點:

缺點:

iframe 嵌套

父應用單獨是一個頁面,每個子應用嵌套一個 iframe,父子通信可採用 postMessage 或者 contentWindow 方式

優點:

缺點:

Web Components

每個子應用需要採用純 Web Components 技術編寫組件,是一套全新的開發模式

優點:

缺點:

webpack5 的 Module Federation

使用 Module Federation,我們可以在一個應用中動態加載並執行另一個應用的代碼,且與技術棧無關,並且能夠共享模塊,從而減小編譯時間以及降低包體積

優點:

缺點:

組合式應用路由分發

每個子應用獨立構建和部署,運行時由父應用來進行路由管理,應用加載,啓動,卸載,以及通信機制

優點:

缺點:

組合式應用路由分發是目前業內普遍使用的一種方案,並且能夠滿足大部分需求,接下來我們詳細來看看這種實現方式

組合式應用路由分發的微前端實現思路

該方案使用的是基座模式,通過一個主應用(基座應用 - Main APP),來管理其它應用(子應用 - MicroAPP)。基座應用大多數是一個前端 SPA 項目,主要負責應用註冊,路由映射,消息下發等,而微應用是獨立前端項目,這些項目不限於採用 React,Vue,Angular 或者 JQuery 開發,每個微應用註冊到基座應用中,由基座進行管理,但是如果脫離基座也是可以單獨訪問

當整個微前端框架運行之後,給用戶的體驗就是類似下圖所示:

簡單描述下就是基座應用中有一些菜單項,點擊每個菜單項可以展示對應的微應用,這些應用的切換是純前端無感知的

上面的實現過程主要如下:

微前端的應用隔離

CSS 隔離

當主應用和微應用同屏渲染時,就可能會有一些樣式會相互污染,可以採取以下兩種方案

而對於微應用與微應用之間的 CSS 隔離就非常簡單,在每次應用加載時,將該應用所有的 link 和 style 內容進行標記。在應用卸載後,同步卸載頁面上對應的 link 和 style 即可

JavaScript 隔離

每當微應用的 JavaScript 被加載並運行時,它的核心實際上是對全局對象 Window 的修改以及一些全局事件的改變,例如 jQuery 這個 js 運行後,會在 Window 上掛載一個 window.$ 對象,對於其他庫 React,Vue 也不例外。爲此,需要在加載和卸載每個微應用的同時,儘可能消除這種衝突和影響,最普遍的做法是採用沙箱機制(SandBox)

沙箱機制的核心是讓局部的 JavaScript 運行時,對外部對象的訪問和修改處在可控的範圍內,即無論內部怎麼運行,都不會影響外部的對象。通常在 Node.js 端可以採用 vm 模塊,而對於瀏覽器,則需要結合 with 關鍵字和 window.Proxy 對象來實現瀏覽器端的沙箱

微前端的消息通信

微前端通常不會限制應用採用的框架,如何在不同的應用,框架之間進行通信是一個需要仔細考量的決定。應用間通信有很多種方式,當然,要讓多個分離的微應用之間要做到通信,本質上仍離不開中間媒介或者說全局對象。

自定義事件

通過事件 [3] 進行通信應該是最簡單、通用的方案了

// 監聽事件
window.addEventListener('message'(event) ={
  // 處理事件
});

// 觸發事件
window.dispatchEvent(new CustomEvent('message'{ detail: input.value }))

發佈 - 訂閱

消息訂閱(pub/sub)模式的通信機制是非常適用的,在基座應用中會定義事件中心 Event,每個微應用分別來註冊事件,當被觸發事件時再有事件中心統一分發,這就構成了基本的通信機制

import { Observable } from 'windowed-observable';

const observable = new Observable('konoha');
observable.subscribe((ninja) ={
  console.log(ninja)
})

observable.publish('Uchiha Shisui');

Web Workers

通過 Web Workers 進行事件通信

import Worky from 'worky'
const worker = new Worky("some-worker.js");

worker.on("eventName"function (some, data) {
  // 處理
});
worker.emit("someEvent", and, some, data);

共享狀態

主應用創建 state store,共享給子應用使用,適用於主、子應用技術棧相同的場景。

總結

微前端是一種將單個巨型應用轉變爲多個微型應用聚合爲一的應用,能夠解決 “巨石應用” 的維護性問題。實現微前端的方式有很多種,每種方案都需要考慮應用隔離和應用通信的問題,目前較爲普遍使用的是組合式路由分發的方式。

參考

參考資料

[1]

TECHNOLOGY RADAR: https://www.thoughtworks.com/zh-cn/radar/techniques/micro-frontends

[2]

import-html-entry: https://www.npmjs.com/package/import-html-entry

[3]

事件: https://developer.mozilla.org/en-US/docs/Web/API/CustomEvent

[4]

微前端(一)- 理念篇: https://www.lumin.tech/blog/micro-frontends-1-concept/

[5]

微前端 - 最容易看懂的微前端知識: https://juejin.cn/post/6844904162509979662

[6]

微前端在解決什麼問題?: https://cloud.tencent.com/developer/article/1546556

[7]

微前端如何落地: https://www.infoq.cn/article/xm_aaiotxmlppgwvx9y9

[8]

微前端解決方案: https://segmentfault.com/a/1190000040275586

[9]

HTML Entry 源碼分析: https://juejin.cn/post/6885212507837825038#heading-6

本文由 Readfog 進行 AMP 轉碼,版權歸原作者所有。
來源https://mp.weixin.qq.com/s/LL6VZj36PKftbwaybBFmXg