你想要的【微前端】都在這裡了!

来源:https://www.cnblogs.com/jingdongkeji/archive/2023/05/04/17370284.html
-Advertisement-
Play Games

某次遇到一個從0到1的大型項目,該項目涉及兩個端,除了鑒權和部分業務邏輯不同外,頁面UI和其餘邏輯幾乎一致,遇到這種項目,該如何架構?既能保證項目順利開發完成,又能保證後期的迭代、維護、可擴展? ...


作者:京東零售 鄭炳懿

開篇:

如果你不知道微前端是什麼,或者不知道微前端能解決什麼問題,那麼你可能不需要微前端。

在我看來,對於每一個沒有使用過的新技術,都應該有以下幾個過程:

1、調研該技術,產出相應的調研文檔。

2、輸出技術Demo,基本的框架結構。

3、試著在項目中使用它,這一步坑會很多。

4、把它推動到線上完成真正的技術升級。

一、調研微前端

1.1 業務背景

某次遇到一個從0到1的大型項目,該項目涉及兩個端,除了鑒權和部分業務邏輯不同外,頁面UI和其餘邏輯幾乎一致,遇到這種項目,該如何架構?既能保證項目順利開發完成,又能保證後期的迭代、維護、可擴展?

1.2 初步方案

首先,想到的技術方案有這麼兩種:

1、復用同一套代碼,通過判斷不同的許可權,服務端下發標識,處理異同的業務邏輯。

2、開發兩套代碼,兩套鑒權各走各的,頁面相同部分從左邊Copy到右邊。

其次,回過頭來想了想,這兩種方案都有缺陷:

1、復用同一套代碼,後期迭代的過程中,業務差異越來越大的時候,就會形成“屎山”。

2、開發兩套代碼,後期迭代的過程中,如果業務依然高度相似,那麼每次都要把A項目中的代碼Copy到B項目中;如果業務逐漸有了各自的風格,那麼兩套代碼的方案顯然是更佳的。

最後,除此之外,還有別的更好的方案嗎?

1.3 什麼是微前端?

微前端的概念是由 ThoughtWorks 在2016年提出的,它是一種前端架構風格,將一個龐大的前端應用拆分成多個獨立靈活的小型應用,每個應用都可以獨立開發、獨立運行、獨立部署,再將這些小型應用融合為一個完整的應用,或者將原本運行已久、沒有關聯的幾個應用融合為一個應用。微前端既可以將多個項目融合為一,又可以減少項目之間的耦合,提高開發效率和可維護性。微前端的核心在於解耦,通過拆分和集成來實現前端應用的可擴展性和靈活性。

圖片來源於micro-app官網

二、初識微前端

2.1 微前端能力

1、獨立開發:微前端可以將一個龐大的前端應用拆分成多個小型應用,每個應用都可以獨立開發,不會影響其他應用的開發進度。

2、獨立部署:每個小型應用都可以獨立部署,不會影響其他應用的部署進度。這也意味著可以使用不同的技術棧、不同的部署方式、不同的版本控制工具等。

3、獨立運行:每個小型應用都可以獨立運行,不會影響其他應用的運行狀態。這也意味著可以使用不同的框架、不同的庫、不同的語言等。

4、集成靈活:微前端框架可以將多個小型應用集成為一個完整的前端應用,或者將原本運行已久、沒有關聯的幾個應用融合為一個應用。這也意味著可以根據需要動態地增加或刪除應用。

5、解耦:微前端可以將前端應用拆分成多個小型應用,每個應用都有自己的職責和業務邏輯,可以減少應用之間的耦合,提高可維護性和可擴展性。

6、增量升級:微前端可以實現增量升級,只需要升級需要更新的小型應用,而不需要升級整個前端應用。這可以減少升級帶來的風險和成本。

2.2 微前端核心

1、拆分:將前端應用拆分成多個小型應用,每個應用都有自己的職責和業務邏輯。這樣可以減少應用之間的耦合,使得每個應用都可以獨立開發、獨立部署和獨立運行。

2、集成:通過微前端框架將多個小型應用集成為一個完整的前端應用。這樣可以根據需要動態地增加或刪除應用,實現靈活的集成。

3、通信:通過定義介面和事件等方式,實現小型應用之間的通信。這樣可以保證各個應用之間的協作和交互,同時又不會影響應用之間的耦合。

4、樣式隔離:通過使用樣式隔離技術,使得每個小型應用都可以使用自己的樣式,不會影響其他應用的樣式。這樣可以保證各個應用之間的樣式不會互相干擾,同時又不會影響應用之間的耦合。

總之,微前端的核心是解耦,通過拆分、集成、通信和樣式隔離等方式,實現前端應用的解耦,提高可維護性和可擴展性。

2.3 微前端平臺

1、single-spa 是一個將多個單頁面應用聚合為一個整體應用的 JavaScript 微前端框架。

2、qiankun 螞蟻金服出品,基於 single-spa 封裝的微前端框架。

3、MicroApp 京東出品,一款基於WebComponent的思想,輕量、高效、功能強大的微前端框架。

由於項目使用的 umi + react +ts 的技術棧,而 qiankun 天生就集成在 umi 框架中了,只需要一些配置就可以使用微前端技術,註意,我這裡說的是一些配置,就是這一些配置,讓我放棄了 qiankun 微前端框架,因為 single-spa 要求子應用修改渲染邏輯並暴露出三個方法:bootstrap、mount、unmount,分別對應初始化、渲染和卸載,這也導致子應用需要對入口文件進行修改。而 qiankun 是基於single-spa進行封裝,所以這些特點也被 qiankun 繼承下來,並且需要對 webpack 配置進行一些修改,成本相對較高。

再來看 micro-app 老東家出品的微前端框架,借鑒了 WebComponent 的思想,通過 CustomElement 結合自定義的 ShadowDom,將微前端封裝成一個類WebComponent 組件,從而實現微前端的組件化渲染。並且由於自定義ShadowDom的隔離特性,micro-app不需要像single-spa和qiankun一樣要求子應用修改渲染邏輯並暴露出方法,也不需要修改 webpack 配置,是目前市面上接入微前端成本最低的方案。

圖片來源於micro-app官網

三、應用微前端

3.1 選擇 mirco-app

結合上述的調研結果,決定使用 micro-app 框架來架構我的這個大型項目。第一,micro-app 使用簡單,學習成本低,小巧的體積和更高的擴展性;第二,老東家的技術,必須全力支持。

確定最終技術方案:

1、項目涉及到兩個端,準備啟用兩個基座,這兩個基座內管理鑒權和統一調用的公共邏輯,基座獨立部署,屬主應用。

2、項目中相同的UI部分,獨立到業務組件庫,可復用,業務邏輯部分,各自在項目中處理,相互獨立。

3.2 啟動 mirco-app

這裡有詳細的使用文檔,就不再贅述,從引入依賴到項目完全渲染出來,只需要四步即可, micro-app 直通車。

值得一提的是第二步,有個小坑,入口處引入包,然後調用方法。

如果你的項目是 Vue的話,這裡說的入口文件應該是 main.js;如果你的項目是 umi 框架的話,入口文件指的的是 src/pages/.umi/umi.js 文件,這個文件是 umi 自動生成的,無法讓你在這裡面編碼,所以你需要在 src 目錄下麵新建一個 index 或者 global 的文件,把下麵的代碼複製進去。

// 項目入口處引入
import microApp from '@micro-zoe/micro-app'

microApp.start()

按照文檔的指引,你應該看到這個界面,如果沒有看到這個界面,那說明你的姿勢有問題,可能是跨域導致,關於跨越問題,Q&A裡面有解決方案,用《程式員的修煉之路》中的一句話來說:“讀一下那些該死的報錯信息”,沒準你就能啟動成功了。

頂部導航和左側菜單是基座,也就是主應用,右側的內容區域,是子應用。

3.3 踩坑 micro-app

3.3.1 路由問題

項目啟動起來要面對的第一個問題就是路由問題。案例裡面給的菜單是一級菜單,但是實際項目應用中可能有二級、甚至三級菜單,那怎麼匹配路由跳轉到對應的子應用呢?

核心代碼:

  microApp.router.push({ 
    name: 'pop',  // 子應用名稱
    path: `${config.pop}${item.url}`  // config是配置文件 item是當前點擊的菜單路徑信息
  });

處理邏輯:

1、優先處理樹形菜單,樹形菜單使用遞歸調用,後期不管是幾級菜單都不用管它了。

2、與服務端約定好樹形菜單的欄位,出必要欄位外,應該包含對應的子應用名稱,路徑,icon圖標等信息,這些信息是你提前給服務端,配置到表結構中的。如果項目足夠大的話,可以啟一個SaaS系統,更加靈活和可靠。

3、當點擊菜單中對應的某個菜單時,取到當前路徑拼接功能變數名稱即可完成跳轉。

 <micro-app
   name="pop"
   url={config?.pop}
 />

3.3.2 麵包屑問題

強烈建議把麵包屑放到子應用中,麵包屑在子應用中的好處是自由完成跳轉,不用主應用做特別的處理,唯一需要處理的是麵包屑裡面的首頁,因為麵包屑放到子應用中,點擊迴首頁時,回到的其實是子應用的首頁,並非是主應用的首頁。

主應用處理邏輯:

import React from 'react';
import config from '@/config';

/** @jsxRuntime classic */
/** @jsx jsxCustomEvent */
import jsxCustomEvent from '@micro-zoe/micro-app/polyfill/jsx-custom-event';

export default ():React.ReactElement => {
  // 子應用點擊了麵包屑的回到首頁
  const onDispathChild = (e:any) => {
    const { isBackHome } = e.detail.data;
    if (isBackHome) window.location.href = '/';
  };

  return (
    <>
      <micro-app
        name="pop"
        url={config?.pop}
        default-page={`${config?.pop}${config.defaultUrl}`}
        onDataChange={onDispathChild}
      />
    </>
  );
};

子應用邏輯:

  // 點擊回到首頁的時候,需要告訴父應用,讓父應用去重置路由
  const onBackHome = () => {
    window.microApp?.dispatch({ isBackHome: true });
  };

micro-app 在 window 下麵掛載了一個全局的對象,我們只需要去觸發它提供的方法,完成主子之間的通信即可,這個邏輯想明白之後,不管是交互邏輯還是數據傳遞邏輯,一通都通。

3.3.3 打包問題

主子應用兩個項目,在進行打包的過程,做了分包的處理,micro-app 中的js沙箱隔離技術有點小缺陷,由於主子應用使用的都是 umi 的框架,打包之後會錯誤的把子應用的包插入主應用中,導致應用報錯,載入不出來。

處理邏輯:

// 不分割組件

dynamicImport: false,

由於在實際項目操作中遇到的問題可能會比以上列舉的比較多,上面舉了幾個典型的例子,後續大家如果使用中遇到什麼問題,也可以私信我進行解決,或者留言評論。

四、總結微前端

最終我們的這個大型項目採用微前端實現了業務解耦,維護性高,擴展性高的期望,後期迭代so easy。

用起來其實還是蠻簡單的,但是用好了不容易,目前我們正在規劃把一整個業務線集成到微前端中,因為有些項目太老了,無法維護了,把這些老項目直接一個鏈接成子應用,新的迭代的都獨立成一個單獨的子應用,可以使用新框架,新技術去實現,技能提高開發效率,又能很好的擴展和迭代,個人覺得微前端技術很優秀,很受用。

以下是一些拆分邏輯,希望給使用微前端技術的同學一些參考:

使用微前端拆分一個大型項目需要註意以下幾點:

1、拆分粒度:應該根據業務功能、團隊職責、技術棧等因素來確定拆分粒度。拆分粒度太小會增加應用之間的通信成本,拆分粒度太大會影響獨立開發和部署的能力。

2、拆分邊界:應該確定每個小型應用的邊界,使得每個應用都有自己的職責和業務邏輯。拆分邊界應該儘可能地減少應用之間的耦合,同時又保證各個應用之間的協作和交互。

3、通信方式:應該確定小型應用之間的通信方式,包括介面、事件等。通信方式應該儘可能地簡單和高效,同時又能夠滿足各個應用之間的協作和交互需求。

4、數據管理:應該確定小型應用之間的數據管理方式,包括數據共用、數據隔離等。數據管理方式應該儘可能地簡單和高效,同時又能夠滿足各個應用之間的數據共用和隔離需求。

5、樣式隔離:應該使用樣式隔離技術,使得每個小型應用都可以使用自己的樣式,不會影響其他應用的樣式。這樣可以保證各個應用之間的樣式不會互相干擾,同時又不會影響應用之間的耦合。

6、集成方式:應該確定集成方式,包括微前端框架的選擇、部署方式等。集成方式應該儘可能地簡單和高效,同時又能夠滿足各個應用之間的集成需求。

總之,使用微前端拆分一個大型項目需要註意拆分粒度、拆分邊界、通信方式、數據管理、樣式隔離和集成方式等方面,以實現前端應用的解耦,提高可維護性和可擴展性。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 如何優雅的將項目中的代碼,亦或是你的demo代碼展示到界面上?本文對使用簡單、便於維護且通用的解決方案,進行相關的對比和探究 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 Symbol是JavaScript中的原始數據類型之一,它表示一個唯一的、不可變的值,通常用作對象屬性的鍵值。由於Symbol值是唯一的,因此可以防止對象屬性被意外地覆蓋或修改。以下是Symbol的方法和屬性整理: 屬性 Symbol.l ...
  • 前言 之前寫了一個vue+django的一個通過串口控制的上位機系統。但是實際生產中,不如部署到伺服器上,這樣可以更好的節約成本。但是這樣就需要弄一個客戶端來控制處理串口信息。那我就在想能不能通過網頁直接拿到客戶端的串口信息。所以問了萬能的chatgpt,得到了以下答案: 是的,前端可以使用 Web ...
  • 在 HTML5 中,文檔對象(即 document 對象)具有一個 visibilityState 屬性,該屬性表示當前文檔對象的可見性狀態。 visibilityState 可能的取值有以下三種: - visible :表示文檔當前處於激活狀態,即當前選項卡處於前臺或當前視窗處於屏幕最上層。- h ...
  • 在隨筆《基於SqlSugar的開發框架循序漸進介紹(28)-- 快速構建系統參數管理界面》中介紹了基於SqlSugar開發框架,構建系統參數管理的後端API部分,以及WInform界面部分內容,本篇隨筆介紹基於Vue3+ElementPlus的前端界面開發過程。 ...
  • 此文為系列文章第一篇,為淺嘗輒止的引入,目的是為了讓前端從業人員及非從業但是對此領域感興趣的人對於”前端“是乾什麼的這個話題有個無門檻的瞭解。 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 1. 原型鏈 溫故而知新: 構造函數、原型和實例的關係: 每個構造函數都有一個原型對象,原型有一個屬性指回構造函數,實例有一個內部指針指向原型。 思考:如果原型是另一個類型的實例呢? 那就意味著這個原型本身有一個內部指針指向另一個原型,相 ...
  • 在HTML中, link 標簽是一個自閉合元素,通常位於文檔的 head 部分。它用於建立與外部資源的關聯,如樣式表、圖標等。 link 標簽具有多個屬性,其中 rel 和 href 是最常用的。 rel 屬性定義了當前文檔與鏈接資源之間的關係。常見的 rel 屬性值有: - stylesheet ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...