QianKun 2.0 微前端源碼解析

此篇文章只簡(jiǎn)單梳理思路拇舀,不會(huì)源碼逐行分析,看此文章前請(qǐng)熟悉qiankun和single-spa的使用最好

暴露出的API, 從index.ts文件可以看出下面暴露的API

export { loadMicroApp, registerMicroApps, start } from './apis';
export { initGlobalState } from './globalState';
export * from './errorHandler';
export * from './effects';
export * from './interfaces';
export { prefetchApps } from './prefetch';

1. 注冊(cè)子應(yīng)用 registerMicroApps

注冊(cè)子應(yīng)用的接口樱拴, 基本原理是內(nèi)部有一個(gè)子應(yīng)用數(shù)組爹土,可以注冊(cè)多個(gè)子應(yīng)用纺且,遍歷數(shù)組調(diào)用single-sparegisterApplication函數(shù)進(jìn)行注冊(cè), 此API用法如下, 可以參考 https://single-spa.js.org/docs/api

registerApplication

注意此函數(shù)的第二個(gè)參數(shù)applicationOrLoadingFn,返回一個(gè)resolved的應(yīng)用或者一個(gè)promise,注意這里的 the resolved application其實(shí)是single-spa定義的一個(gè)概念妻往,一個(gè)應(yīng)用的概念,具體參考https://single-spa.js.org/docs/building-applications
說(shuō)白了就是要返回一個(gè)如下的對(duì)象试和,帶有名字讯泣,聲明周期方法, 這就是qiankun要求你子應(yīng)用必須聲明這些聲明周期函數(shù)的目的,返回給single-spa

{
  name: "applicationName",
  bootstrap: [ bootstrap function array],
  mount: [ mount function array],
  unmount: [ unmount function array]
}

registerMicroApps里面關(guān)鍵的代碼如下阅悍,注意這個(gè)loadApp才是真正封裝的返回應(yīng)用的地方

  unregisteredApps.forEach(app => {
    const { name, activeRule, props, ...appConfig } = app;

    registerApplication({
      name,
      app: async () => {
        await frameworkStartedDefer.promise;
        return loadApp({ name, props, ...appConfig }, frameworkConfiguration, lifeCycles);
      },
      activeWhen: activeRule,
      customProps: props,
    });
  });

加載子應(yīng)用內(nèi)部函數(shù) loadApp

下面著重看下這個(gè)加載子應(yīng)用函數(shù)好渠,分幾個(gè)小步驟看下

  1. 加載入口文件 importEntry: 加載指定的html文本,和腳本用來(lái)執(zhí)行节视,這就是你在qiankun中傳入entry: '//localhost:7100'類似參數(shù)的用途拳锚,用此路勁來(lái)加載你子應(yīng)用的入口文件,和手動(dòng)執(zhí)行html中的腳本
  2. 創(chuàng)建子應(yīng)用DOM render: 如果你注冊(cè)的時(shí)候加入了render方法,此處會(huì)調(diào)用執(zhí)行寻行,否則找到container元素霍掺,把渲染的內(nèi)容插入節(jié)點(diǎn)中
  3. 創(chuàng)建沙箱 sandbox: 此處是qiankun的核心,創(chuàng)建了js的沙箱進(jìn)行子應(yīng)用之間和主應(yīng)用的隔離, 此處不展開(kāi)分析拌蜘,原理就是window對(duì)象的屬性劫持杆烁,內(nèi)部用一個(gè)對(duì)象緩存子應(yīng)用window下面屬性方法,達(dá)到隔離的目的简卧,下面代碼可以看出兔魂,支持proxy的就用proxy代理的沙箱隔離,否則用SnapshotSandbox举娩,簡(jiǎn)單來(lái)說(shuō)就是內(nèi)部備份析校,diff 方式記錄, SingularProxySandbox是老沙箱方式實(shí)現(xiàn)铜涉,說(shuō)是穩(wěn)定了再用統(tǒng)一的ProxySandbox
  if (window.Proxy) {
    sandbox = singular ? new LegacySandbox(appName) : new ProxySandbox(appName);
  } else {
    sandbox = new SnapshotSandbox(appName);
  }
  1. 執(zhí)行加載前事件 beforeLoad: 利用execHooksChain函數(shù)歸并執(zhí)行函數(shù)智玻,簡(jiǎn)單來(lái)說(shuō)就是類似redux中的compose洋蔥模型,上一次函數(shù)的返回值作為下一個(gè)函數(shù)的參數(shù)骄噪,包裹執(zhí)行尚困,也是中間件原理,后面的一些加載聲明周期函數(shù)與此類似
  2. 執(zhí)行入口文件的腳本 execScripts: 此步驟就是獲取子應(yīng)用指定的主腳本链蕊,執(zhí)行并且獲取子應(yīng)用的bootstrap,mount, unmount等聲明周期函數(shù)
  3. 返回應(yīng)用并執(zhí)行內(nèi)部的一些鉤子周期函數(shù): 此時(shí)就可以返回一個(gè)滿足single-spa的應(yīng)用了事甜,同時(shí)在mount和unmount之前會(huì)執(zhí)行一些內(nèi)部定義的生命周期鉤子

到此為止,基本整個(gè)注冊(cè)滔韵,加載流程清楚了

2. 動(dòng)態(tài)加載應(yīng)用 loadMicroApp

qiankun提供了動(dòng)態(tài)加載子應(yīng)用的方式逻谦,簡(jiǎn)單代碼如下, 動(dòng)態(tài)調(diào)用了single-spamountRootParcel方法和qiankun內(nèi)部loadApp來(lái)實(shí)現(xiàn)應(yīng)用加載

export function loadMicroApp<T extends object = {}>(
  app: LoadableApp<T>,
  configuration = frameworkConfiguration,
): MicroApp {
  const { props, ...appConfig } = app;
  return mountRootParcel(() => loadApp(appConfig, configuration), {
    domElement: document.createElement('div'),
    ...props,
  });
}

3. 啟動(dòng)應(yīng)用 start

start的代碼也沒(méi)有幾行,如下主要就是調(diào)用single-spa內(nèi)部的start來(lái)啟動(dòng)應(yīng)用陪蜻,對(duì)應(yīng)的為startSingleSpa此函數(shù)

export function start(opts: FrameworkConfiguration = {}) {
  frameworkConfiguration = opts;
  const {
    prefetch = true,
    sandbox = true,
    singular = true,
    urlRerouteOnly,
    ...importEntryOpts
  } = frameworkConfiguration;

  if (prefetch) {
    prefetchApps(microApps, prefetch, importEntryOpts);
  }

  if (sandbox) {
    if (!window.Proxy) {
      console.warn('[qiankun] Miss window.Proxy, proxySandbox will degenerate into snapshotSandbox');
      // 快照沙箱不支持非 singular 模式
      if (!singular) {
        console.error('[qiankun] singular is forced to be true when sandbox enable but proxySandbox unavailable');
        frameworkConfiguration.singular = true;
      }
    }
  }

  startSingleSpa({ urlRerouteOnly });

  frameworkStartedDefer.resolve();
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末邦马,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌滋将,老刑警劉巖邻悬,帶你破解...
    沈念sama閱讀 221,820評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異随闽,居然都是意外死亡父丰,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,648評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門掘宪,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)蛾扇,“玉大人,你說(shuō)我怎么就攤上這事魏滚《剖祝” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 168,324評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵鼠次,是天一觀的道長(zhǎng)更哄。 經(jīng)常有香客問(wèn)我,道長(zhǎng)须眷,這世上最難降的妖魔是什么竖瘾? 我笑而不...
    開(kāi)封第一講書人閱讀 59,714評(píng)論 1 297
  • 正文 為了忘掉前任,我火速辦了婚禮花颗,結(jié)果婚禮上捕传,老公的妹妹穿的比我還像新娘。我一直安慰自己扩劝,他們只是感情好庸论,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,724評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著棒呛,像睡著了一般聂示。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上簇秒,一...
    開(kāi)封第一講書人閱讀 52,328評(píng)論 1 310
  • 那天鱼喉,我揣著相機(jī)與錄音,去河邊找鬼趋观。 笑死扛禽,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的皱坛。 我是一名探鬼主播编曼,決...
    沈念sama閱讀 40,897評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼剩辟!你這毒婦竟也來(lái)了掐场?” 一聲冷哼從身側(cè)響起往扔,我...
    開(kāi)封第一講書人閱讀 39,804評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎熊户,沒(méi)想到半個(gè)月后萍膛,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,345評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡嚷堡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,431評(píng)論 3 340
  • 正文 我和宋清朗相戀三年卦羡,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片麦到。...
    茶點(diǎn)故事閱讀 40,561評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖欠肾,靈堂內(nèi)的尸體忽然破棺而出瓶颠,到底是詐尸還是另有隱情,我是刑警寧澤刺桃,帶...
    沈念sama閱讀 36,238評(píng)論 5 350
  • 正文 年R本政府宣布粹淋,位于F島的核電站,受9級(jí)特大地震影響瑟慈,放射性物質(zhì)發(fā)生泄漏桃移。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,928評(píng)論 3 334
  • 文/蒙蒙 一葛碧、第九天 我趴在偏房一處隱蔽的房頂上張望借杰。 院中可真熱鬧,春花似錦进泼、人聲如沸蔗衡。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 32,417評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)绞惦。三九已至,卻和暖如春洋措,著一層夾襖步出監(jiān)牢的瞬間济蝉,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 33,528評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工菠发, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留王滤,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,983評(píng)論 3 376
  • 正文 我出身青樓雷酪,卻偏偏與公主長(zhǎng)得像淑仆,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子哥力,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,573評(píng)論 2 359