十一、Next.js,延遲加載模塊

Next.js是一個新的通用JavaScript框架,它為基于React和服務器的Web應用提供了一個新的可選方案。

Next.js目前已經開源,https://zeit.co/blog/next

Next.js會自動進行代碼分割,它基于你的應用程序中的頁面。例如,如果你的一個模塊至少在你的頁面的一半使用,然后它進入到主JavaScript包中。如果沒有,該模塊將停留在頁面的包中。

這是一個相當不錯的默認設置。但有時,我們需要更好地控制加載模塊。例如,看看下面的場景:

  • 我們正在建立一個基于official firebase API,的hacker news clone
  • 我們在服務器上獲取數據來執行SSR,
  • 當需要時,我們還會在客戶端獲取數據(當切換頁面)

:在本例中,我們的主應用程序包包含了 firebase 模塊,因為它在我們的所有頁面中都使用過。這是一個相當大的模塊。(超過了react,react-dom和next.js合在一起的大小 )

但當到了客戶端時,我們只是在當用戶開始瀏覽不同的頁面時,需要它。因此,如果我們能夠在那個時候加載firebase模塊,我們就可以改進我們的應用程序的初始加載。

這正是我們在這堂課上要做的。

安裝

在這節課上,我們已經建立了一個非常基礎的hacker news clone。
下面是如何得到它的方法:

git clone https://github.com/arunoda/learnnextjs-demo.git
cd learnnextjs-demo
git checkout firebase-hn

然后你就可以運行這個應用了:

npm install
npm run dev

現在,訪問http://localhost:3000并嘗試應用程序。

分析

這款應用看起來是這樣的,它是一個非常基本的功能:

Paste_Image.png

現在,讓我們試著了解一下我們的應用程序包里面是什么。

為此,只需運行以下命令:

npm run analyze

然后,它將啟動一個 webpack bundle analyzer 分析器,您將能夠檢查每個JavaScript包內部的內容。

因此,firebase模塊包含在哪里呢?

  • inside commons.js bundle
  • inside main.js bundle
  • inside pages/index.js bundle
  • in all pages
分析結果

正如您所看到的,firebase模塊保留在commons.js包。

Paste_Image.png

這很簡單,因為它在我們應用的所有頁面中都使用過。

延遲加載

只有當用戶試圖導航到不同的頁面時,我們才使用firebase模塊。所以,如果我們能在那個時候加載firebase模塊,這對我們的應用來說是一個巨大的勝利。

幸運的是,我們可以很容易地做到這一點通過Next.js的動態導入功能。

讓我們開始吧。

與firebase相關的代碼位于lib/load-db.js文件中。這里是內容:

export default async () => {
  const firebase = require('firebase')

  try {
    firebase.initializeApp({
      databaseURL: 'https://hacker-news.firebaseio.com'
    })
  } catch (err) {
    // we skip the "already exists" message which is
    // not an actual error when we're hot-reloading
    if (!/already exists/.test(err.message)) {
      console.error('Firebase initialization error', err.stack)
    }
  }

  return firebase.database().ref('v0')
}

這段代碼在每個頁面的getInitialProps函數中使用。
這是一個相當不錯的代碼,它使用require 加載firebase 模塊。

現在,我們要對上面的代碼做一個小的改動,當我們需要firebase模塊時。

// const firebase = require('firebase')
const firebase = await import('firebase')

在這里,我們使用import()函數來加載firebase模塊。它返回一個promise,我們使用await并resolve這個模塊。

嘗試應用上述更改并再次分析JavaScript包:

npm run analyze

然后,選擇firebase模塊所在的包的名稱。這可能是:

  • commons.js
  • main.js
  • chunks/firebase.js
  • chunks/firebase-[a-random-string].js
自己的包

正如你看到的,它有自己的bundle,它的名字看起來像:

chunks/firebase-[a-random-string].js
Paste_Image.png

當您試圖導入firebase時,這個包就被加載了。

讓我們做個測試

現在讓我們試著看看它在瀏覽器中是如何工作的。

為此,我們需要運行我們的應用程序的生產版本,你可以這樣做:

npm run build
npm run start

然后,在瀏覽器中啟動該應用程序,該程序具有良好的網絡檢查調試器。(為了讓事情變得簡單,我建議你應該使用Chrome)

現在,在Chrome中加載http://localhost:3000并打開網絡檢查器。

Paste_Image.png

然后,清除網絡檢查器中的當前數據。

你可以在上面的圖片中點擊紅色方塊的選擇圖標。
但是,如果瀏覽器版本發生了變化,那么位置可能就在不同的位置。

現在,單擊頁面上列出的任何標題。檢查網絡督察。
然后,點擊“Home”鏈接,再次進入首頁。檢查網絡督察。

你如何最好地描述你所檢查過的東西?

  • “firebase” bundle loads every time
  • “firebase” bundle loads only in the first time
  • “firebase” bundle loads only in the second time
  • “firebase” bundle never loads
測試結果

正如您所見,它只在您第一次瀏覽頁面時才加載。這就是實際發生的情況。

在第一次,pages/post.js 頁面的 getIntitialProps 導入firebase模塊(通過lib/load-db.js)。所以,這個應用程序加載了這個包。

即使是第二次,pages/index.js 頁面導入 firebase 模塊,但是在那個時候,它已經被加載了,并且沒有理由再加載它。

最后

坦率地說,這個示例并不是延遲加載的最佳用例。只是因為,

  • 您需要在所有頁面中使用firebase模塊。
  • 延遲加載的firebase模塊減小了主JavaScript包app.js的大小。但是它不會影響頁面加載時間,因為頁面是由服務器呈現的。
  • 主JavaScript包的加載不會阻塞初始的HTML渲染。

這給我們帶來的唯一好處是快速的JavaScript交互,因為應用程序.js由于減小的大小而載入速度更快。

無論如何,這是一個很好的例子,我們可以演示延遲加載模塊。

因此,你可以在你的應用中使用它。

本文翻譯自:https://learnnextjs.com/excel/lazy-loading-modules

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,837評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,196評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,688評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,654評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,456評論 6 406
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,955評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,044評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,195評論 0 287
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,725評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,608評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,802評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,318評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,048評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,422評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,673評論 1 281
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,424評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,762評論 2 372

推薦閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,605評論 25 708
  • 那是一片平靜的海。覆蓋著無窮的海面慢慢地波動著,不愿在表面激起一絲浪花。海邊黑色的礁石被海波敲著,時而被淹沒,時而...
    楠路閱讀 261評論 0 0
  • 題記:每個人都有灰頭土臉的往事,重要的是如何拍落衣襟上的塵土,和往事說再見,在飛揚的迷茫中撥云見霧找尋陽光,跟著太...
    許二可閱讀 301評論 0 5
  • 2015.11.14 期盼的陽光明媚,結果陰雨綿綿,但也無妨。 天氣一轉冷,早上行走的力量會降低很多,坐上714,...
    茗小麥閱讀 261評論 3 1