Angular HMR(熱模塊替換) 功能實(shí)現(xiàn)

最近一個(gè)同事在使用Angular的時(shí)候,希望能像VUE那樣,修改代碼后瀏覽器不刷新,頁面對(duì)應(yīng)修改的組件自動(dòng)更新的功能。這個(gè)功能的名字時(shí)HMR (hot module replace)。
稍微研究了一下,發(fā)現(xiàn)在angular/cli創(chuàng)建的項(xiàng)目中,實(shí)現(xiàn)這個(gè)不算太難,步驟如下:

  1. 首先創(chuàng)建一個(gè)src/environments/environment.hmr.ts文件,內(nèi)容如下
export const environment = {
 production: false,
 hmr: true
};

當(dāng)然,對(duì)應(yīng)的environment.prod.tsenvironment.ts需要增加一個(gè)hmr:false.

如果environment.ts里面的hmr設(shè)置為ture,那么ng serve --hmr也有同樣效果。不過我對(duì)熱替換功能還不是那么相信,重新刷新瀏覽器能保證狀態(tài)更加干凈一些,所以讓environment.ts中的hmrfalse.

  1. .angular-cli.json文件的環(huán)境中增加hmr的環(huán)境,大致如下:
"environments": {
  ...
  "hmr": "environments/environment.hmr.ts",
}
  1. package.jsonscripts中增加一個(gè)新的命令。(當(dāng)然也可以不增加,直接運(yùn)行ng serve --hmr -e=hmr和運(yùn)行npm run hmr效果一樣)
"scripts": {
  ...
  "hmr": "ng serve --hmr -e=hmr"
}
  1. 安裝hmr模塊,命令如下:
    npm install --save-dev @angularclass/hmr

  2. 創(chuàng)建src\hmr.ts文件,內(nèi)容如下:

import { NgModuleRef, ApplicationRef } from '@angular/core';
import { createNewHosts } from '@angularclass/hmr';

export const hmrBootstrap = (module: any, bootstrap: () => Promise<NgModuleRef<any>>) => {
  let ngModule: NgModuleRef<any>;
  module.hot.accept();
  bootstrap().then(currentModule => ngModule = currentModule);
  module.hot.dispose(() => {
    const appRef: ApplicationRef = ngModule.injector.get(ApplicationRef);
    const elements = appRef.components.map(c => c.location.nativeElement);
    const removeOldHosts = createNewHosts(elements);
    ngModule.destroy();
    removeOldHosts();
  });
};

這事熱替換的關(guān)鍵,hmrBootstrap會(huì)替換原始的bootstrap(下面會(huì)看到), 替換后,當(dāng)有新的模塊更新時(shí),hmr會(huì)首先移除掉舊有的模塊,然后接收新的模塊。這些都是發(fā)生在瀏覽器里面。所以頁面不會(huì)刷新。

  1. 更新src\main.ts文件如下:
import { enableProdMode } from '@angular/core';
import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';

import { AppModule } from './app/app.module';
import { environment } from './environments/environment';

import { hmrBootstrap } from './hmr';

if (environment.production) {
  enableProdMode();
}

const bootstrap = () => platformBrowserDynamic().bootstrapModule(AppModule);

if (environment.hmr) {
  if (module[ 'hot' ]) {
    hmrBootstrap(module, bootstrap);
  } else {
    console.error('Ammm.. HMR is not enabled for webpack');
  }
} else {
  bootstrap();
}

這里替換就得啟動(dòng)文件,如果設(shè)置為hmr,那么調(diào)用hmrBootStrap來啟動(dòng)網(wǎng)頁,否則就用過去的

  1. 現(xiàn)在運(yùn)行npm run hmr或者ng serve --hmr -e=hmr,就實(shí)現(xiàn)了熱替換功能。

參考文獻(xiàn): https://medium.com/wizardnet972/hot-module-replacement-with-angular-cli-5fc7a3ae4a9c

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容