作者:京東零售 鄭炳懿
開篇:
如果你不知道微前端是什麼,或者不知道微前端能解決什麼問題,那麼你可能不需要微前端。
在我看來,對於每一個沒有使用過的新技術,都應該有以下幾個過程:
1、調研該技術,產出相應的調研檔案。
2、輸出技術Demo,基本的框架結構。
3、試著在專案中使用它,這一步坑會很多。
4、把它推動到線上完成真正的技術升級。
某次遇到一個從0到1的大型專案,該專案涉及兩個端,除了鑑權和部分業務邏輯不同外,頁面UI和其餘邏輯幾乎一致,遇到這種專案,該如何架構?既能保證專案順利開發完成,又能保證後期的迭代、維護、可延伸?
首先,想到的技術方案有這麼兩種:
1、複用同一套程式碼,通過判斷不同的許可權,伺服器端下發標識,處理異同的業務邏輯。
2、開發兩套程式碼,兩套鑑權各走各的,頁面相同部分從左邊Copy到右邊。
其次,回過頭來想了想,這兩種方案都有缺陷:
1、複用同一套程式碼,後期迭代的過程中,業務差異越來越大的時候,就會形成「屎山」。
2、開發兩套程式碼,後期迭代的過程中,如果業務依然高度相似,那麼每次都要把A專案中的程式碼Copy到B專案中;如果業務逐漸有了各自的風格,那麼兩套程式碼的方案顯然是更佳的。
最後,除此之外,還有別的更好的方案嗎?
微前端的概念是由 ThoughtWorks 在2016年提出的,它是一種前端架構風格,將一個龐大的前端應用拆分成多個獨立靈活的小型應用,每個應用都可以獨立開發、獨立執行、獨立部署,再將這些小型應用融合為一個完整的應用,或者將原本執行已久、沒有關聯的幾個應用融合為一個應用。微前端既可以將多個專案融合為一,又可以減少專案之間的耦合,提高開發效率和可維護性。微前端的核心在於解耦,通過拆分和整合來實現前端應用的可延伸性和靈活性。
圖片來源於micro-app官網
1、獨立開發:微前端可以將一個龐大的前端應用拆分成多個小型應用,每個應用都可以獨立開發,不會影響其他應用的開發進度。
2、獨立部署:每個小型應用都可以獨立部署,不會影響其他應用的部署進度。這也意味著可以使用不同的技術棧、不同的部署方式、不同的版本控制工具等。
3、獨立執行:每個小型應用都可以獨立執行,不會影響其他應用的執行狀態。這也意味著可以使用不同的框架、不同的庫、不同的語言等。
4、整合靈活:微前端框架可以將多個小型應用整合為一個完整的前端應用,或者將原本執行已久、沒有關聯的幾個應用融合為一個應用。這也意味著可以根據需要動態地增加或刪除應用。
5、解耦:微前端可以將前端應用拆分成多個小型應用,每個應用都有自己的職責和業務邏輯,可以減少應用之間的耦合,提高可維護性和可延伸性。
6、增量升級:微前端可以實現增量升級,只需要升級需要更新的小型應用,而不需要升級整個前端應用。這可以減少升級帶來的風險和成本。
1、拆分:將前端應用拆分成多個小型應用,每個應用都有自己的職責和業務邏輯。這樣可以減少應用之間的耦合,使得每個應用都可以獨立開發、獨立部署和獨立執行。
2、整合:通過微前端框架將多個小型應用整合為一個完整的前端應用。這樣可以根據需要動態地增加或刪除應用,實現靈活的整合。
3、通訊:通過定義介面和事件等方式,實現小型應用之間的通訊。這樣可以保證各個應用之間的共同作業和互動,同時又不會影響應用之間的耦合。
4、樣式隔離:通過使用樣式隔離技術,使得每個小型應用都可以使用自己的樣式,不會影響其他應用的樣式。這樣可以保證各個應用之間的樣式不會互相干擾,同時又不會影響應用之間的耦合。
總之,微前端的核心是解耦,通過拆分、整合、通訊和樣式隔離等方式,實現前端應用的解耦,提高可維護性和可延伸性。
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官網
結合上述的調研結果,決定使用 micro-app 框架來架構我的這個大型專案。第一,micro-app 使用簡單,學習成本低,小巧的體積和更高的擴充套件性;第二,老東家的技術,必須全力支援。
確定最終技術方案:
1、專案涉及到兩個端,準備啟用兩個基座,這兩個基座內管理鑑權和統一呼叫的公共邏輯,基座獨立部署,屬主應用。
2、專案中相同的UI部分,獨立到業務元件庫,可複用,業務邏輯部分,各自在專案中處理,相互獨立。
這裡有詳細的使用檔案,就不再贅述,從引入依賴到專案完全渲染出來,只需要四步即可, 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裡面有解決方案,用《程式設計師的修煉之路》中的一句話來說:「讀一下那些該死的報錯資訊」,沒準你就能啟動成功了。
頂部導航和左側選單是基座,也就是主應用,右側的內容區域,是子應用。
專案啟動起來要面對的第一個問題就是路由問題。案例裡面給的選單是一級選單,但是實際專案應用中可能有二級、甚至三級選單,那怎麼匹配路由跳轉到對應的子應用呢?
核心程式碼:
microApp.router.push({
name: 'pop', // 子應用名稱
path: `${config.pop}${item.url}` // config是組態檔 item是當前點選的選單路徑資訊
});
處理邏輯:
1、優先處理樹形選單,樹形選單使用遞迴呼叫,後期不管是幾級選單都不用管它了。
2、與伺服器端約定好樹形選單的欄位,出必要欄位外,應該包含對應的子應用名稱,路徑,icon圖示等資訊,這些資訊是你提前給伺服器端,設定到表結構中的。如果專案足夠大的話,可以啟一個SaaS系統,更加靈活和可靠。
3、當點選選單中對應的某個選單時,取到當前路徑拼接域名即可完成跳轉。
<micro-app
name="pop"
url={config?.pop}
/>
強烈建議把麵包屑放到子應用中,麵包屑在子應用中的好處是自由完成跳轉,不用主應用做特別的處理,唯一需要處理的是麵包屑裡面的首頁,因為麵包屑放到子應用中,點選回首頁時,回到的其實是子應用的首頁,並非是主應用的首頁。
主應用處理邏輯:
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 下面掛載了一個全域性的物件,我們只需要去觸發它提供的方法,完成主子之間的通訊即可,這個邏輯想明白之後,不管是互動邏輯還是資料傳遞邏輯,一通都通。
主子應用兩個專案,在進行打包的過程,做了分包的處理,micro-app 中的js沙箱隔離技術有點小缺陷,由於主子應用使用的都是 umi 的框架,打包之後會錯誤的把子應用的包插入主應用中,導致應用報錯,載入不出來。
處理邏輯:
// 不分割元件
dynamicImport: false,
由於在實際專案操作中遇到的問題可能會比以上列舉的比較多,上面舉了幾個典型的例子,後續大家如果使用中遇到什麼問題,也可以私信我進行解決,或者留言評論。
最終我們的這個大型專案採用微前端實現了業務解耦,維護性高,擴充套件性高的期望,後期迭代so easy。
用起來其實還是蠻簡單的,但是用好了不容易,目前我們正在規劃把一整個業務線整合到微前端中,因為有些專案太老了,無法維護了,把這些老專案直接一個連結成子應用,新的迭代的都獨立成一個單獨的子應用,可以使用新框架,新技術去實現,技能提高開發效率,又能很好的擴充套件和迭代,個人覺得微前端技術很優秀,很受用。
以下是一些拆分邏輯,希望給使用微前端技術的同學一些參考:
使用微前端拆分一個大型專案需要注意以下幾點:
1、拆分粒度:應該根據業務功能、團隊職責、技術棧等因素來確定拆分粒度。拆分粒度太小會增加應用之間的通訊成本,拆分粒度太大會影響獨立開發和部署的能力。
2、拆分邊界:應該確定每個小型應用的邊界,使得每個應用都有自己的職責和業務邏輯。拆分邊界應該儘可能地減少應用之間的耦合,同時又保證各個應用之間的共同作業和互動。
3、通訊方式:應該確定小型應用之間的通訊方式,包括介面、事件等。通訊方式應該儘可能地簡單和高效,同時又能夠滿足各個應用之間的共同作業和互動需求。
4、資料管理:應該確定小型應用之間的資料管理方式,包括資料共用、資料隔離等。資料管理方式應該儘可能地簡單和高效,同時又能夠滿足各個應用之間的資料共用和隔離需求。
5、樣式隔離:應該使用樣式隔離技術,使得每個小型應用都可以使用自己的樣式,不會影響其他應用的樣式。這樣可以保證各個應用之間的樣式不會互相干擾,同時又不會影響應用之間的耦合。
6、整合方式:應該確定整合方式,包括微前端框架的選擇、部署方式等。整合方式應該儘可能地簡單和高效,同時又能夠滿足各個應用之間的整合需求。
總之,使用微前端拆分一個大型專案需要注意拆分粒度、拆分邊界、通訊方式、資料管理、樣式隔離和整合方式等方面,以實現前端應用的解耦,提高可維護性和可延伸性。