esbuild:一款快 10-100 倍的 JS 打包 / 壓縮工具

為什麼又造個輪子?

為什麼又要構建一個 JavaScript 構建工具呢?因為當前用於 Web 的構建工具比用戶期望的性能至少慢一個數量級。我希望這個項目可以作為一種“存在證明”,證明我們的 JavaScript 工具實際上能比現在快得多。

基準測試

我想到的用例是打包用於生產的大型代碼庫。這個流程包括壓縮代碼以減少網絡傳輸時間,以及生成源映射(對於調試生產中的錯誤是非常重要的)。理想情況下,構建工具還應該具備快速構建能力,而不必先預熱緩存。

我的主基準測試會將 three.js 庫複製 10 次並從頭開始構建單個包,過程中沒有任何緩存,從而模擬一個大型代碼庫。在這個基準測試中,esbuild 比我測試的其他 JavaScript 打包器(Webpack、Rollup、Parcel 和 FuseBox)快 10-100 倍。這個基準測試可以使用’make bench-three’來運行。

esbuild:一款快 10-100 倍的 JS 打包 / 壓縮工具

esbuild:一款快 10-100 倍的 JS 打包 / 壓縮工具

時間數據取三次運行中最好的一次,主要運行環境如下:

  • 使用’–bundle --minify --sourcemap’來運行 esbuild。
  • 使用’rollup-plugin-terser’插件,因為 rollup 自身不支持壓縮。
  • Webpack 使用的是’–mode = production --devtool = sourcemap’。
  • Parcel 使用默認選項。
  • FuseBox 使用’useSingleBundle: true’配置。
  • 絕對速度基於總行數(包括註釋和空白行),當前為 547,441。
  • 測試是在配備 16GB RAM 的 6 核 2019 MacBook Pro 上完成的。

為什麼這麼快?

幾個原因:

  • 它是用 Go 語言編寫的,該語言可以編譯為原生代碼;
  • 解析,打印和源映射生成全部完全並行化;
  • 無需昂貴的數據轉換,只需很少的幾步即可完成所有操作;
  • 編寫代碼時處處注意速度表現,並儘量避免不必要的配置。

狀態

目前支持:

  • CommonJS 模塊
  • ES6 模塊
  • 使用’–bundle’與 ES6 模塊的靜態綁定打包
  • 使用’–minify’完全壓縮(空格、標識符和修飾符)
  • 啟用’–sourcemap’時,完全支持源映射
  • .jsx 文件的 JSX 到 JavaScript 轉換
  • 通過’–define’進行編譯時標識符替換
  • 使用 package.json 中的’browser’字段進行路徑替換
  • 自動檢測 tsconfig.json 中的’baseUrl’

這是我在 2019-2020 年寒假期間寫的一項業餘愛好項目。我相信它是相對完整和實用的。但它是全新的代碼,可能有很多錯誤。還沒有任何人在生產中使用過它。使用風險自負。

還請記住,它並不完全支持將現代語言語法降低到早期語言版本。目前只有類字段和 nullish 合併運算符是支持的。

我個人不想運營一個大型的開源項目,因此我目前沒在尋求貢獻。

安裝

如果你已安裝 Go 語言工具鏈,可以使用’make’生成可執行文件。當前可在 npm 上的單獨軟件包中找到預構建的二進制文件:

<code>npm install -g esbuild-linux-64   # for Linuxnpm install -g esbuild-darwin-64  # for macOSnpm install -g esbuild-windows-64 # for Windowsnpm install -g esbuild-wasm       # for all other platforms/<code>

這將添加一個名為’esbuild’的命令。

用法

命令行界面獲取入口點列表,併為每個入口點生成一個打包文件。以下是可用的選項:

複製代碼

<code>Usage:  esbuild [options] [entry points] Options:  --name=...            模塊名稱  --bundle               將所以依賴項打包進輸出文件  --outfile=...         輸出文件 (用於一個入口點)  --outdir=...          輸出目錄 (用於多個入口點)  --sourcemap            發出一個源映射  --error-limit=...     最大錯誤計數,0 是禁用 (默認值為 10)  --target=...          語言目標 (默認 esnext)   --minify              設置所有 --minify-* flags  --minify-whitespace   移除空格  --minify-identifiers  縮短標識符  --minify-syntax       使用較短的等效語法   --define:K=V          解析時用 V 替換 K  --jsx-factory=...     用來替換 React.createElement 的內容  --jsx-fragment=...    用來替換 React.Fragment 的內容   --trace=...           在這個文件中寫入一個 CPU trace  --cpuprofile=...     在這個文件中寫入一個 CPU profileExample:  # Produces dist/entry_point.js and dist/entry_point.js.map  esbuild --bundle entry_point.js --outdir=dist --minify --sourcemap/<code>

與 React 一起使用

要將 esbuild 與 React 一起使用:

  • 確保所有 JSX 語法都放在.jsx 文件而不是.js 文件中,因為 esbuild 使用文件擴展名來確定要解析的語法。
  • 如果你使用的是 TypeScript,請先運行’tsc’以將.tsx 文件轉換為.jsx 或.js 文件。
  • 如果你使用 esbuild 打包 React 自身,而不是在 HTML 中使用


分享到:


相關文章: