欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Webpack打包慢問題的完美解決方法

 更新時間:2017年03月16日 09:37:37   作者:王偉嘉  
這篇文章主要給大家介紹了關(guān)于Webpack打包慢問題的完美解決方法,文中通過示例代碼介紹的非常詳細,相信對大家的學習或者工作能帶來一定的幫助,需要的朋友們下面來一起看看吧。

前言

這幾天寫騰訊實習生 Mini 項目的時候用上了 React 全家桶,當然同時引入了 Webpack 作為打包工具。但是開發(fā)過程中遇到一個很棘手的問題就是,React 加上 React-Router、superagent、eventproxy 這些第三方輪子一共有好幾百個 module,Webpack 的打包速度極慢。這對于開發(fā)是非常不好的體驗,同時效率也極低。

問題分析

我們先來看一下完全沒有任何優(yōu)化的時候,Webpack 的打包速度(使用了jsx和babel的loader)。

下面是我們的測試文件:

//test.js
var react = require('react');
var ReactAddonsCssTransitionGroup = require('react-addons-css-transition-group');
var reactDOM = require('react-dom');
var reactRouter = require('react-router');
var superagent = require("superagent");
var eventproxy = require("eventproxy");

運行

webpack test.js

在我的2015款RMBP13,i5處理器,全SSD下,性能是這樣的:

沒錯你沒有看錯,這幾個第三方輪子加起來有整整668個模塊,全部打包需要20多秒。

這意味著什么呢?你每次對業(yè)務代碼的修改,gulp 或者 Webpack 監(jiān)測到后都會重新打包,你要足足等20秒才能看到自己的修改結(jié)果。

但是需要重新打包的只有你的業(yè)務代碼,這些第三方庫是完全不用重新打包的,它們的存在只會拖累打包性能。所以我們要找一些方法來優(yōu)化這個過程。

配置externals

Webpack 可以配置 externals 來將依賴的庫指向全局變量,從而不再打包這個庫,比如對于這樣一個文件:

import React from 'react';
console.log(React);

如果你在 Webpack.config.js 中配置了externals:

module.exports = {
 externals: {
  'react': 'window.React'
 }
 //其它配置忽略...... 
};

等于讓 Webpack 知道,對于 react 這個模塊就不要打包啦,直接指向 window.React 就好。不過別忘了加載 react.min.js,讓全局中有 React 這個變量。

我們來看看性能,因為不用打包 React 了所以速度自然超級快,包也很?。?/p>

配置externals的缺陷

問題如果就這么簡單地解決了的話,那我就沒必要寫這篇文章了,下面我們加一個 react 的動畫庫 react-addons-css-transition-group 來試一試:

import React from 'react';
import ReactAddonsCssTransitionGroup from 'react-addons-css-transition-group';
console.log(React);

對,你沒有看錯,我也沒有截錯圖,新加了一個很小很小的動畫庫之后,性能又爆炸了。從模塊數(shù)來看,一定是 Webpack 又把 react 重新打包了一遍。

我們來看一下為什么一個很小很小的動畫庫會導致 Webpack 又傻傻地把 react 重新打包了一遍。找到 react-addons-css-transition-group 這個模塊,然后看看它是怎么寫的:

// react-addons-css-transition-group模塊
// 入口文件 index.js
module.exports = require('react/lib/ReactCSSTransitionGroup');

這個動畫模塊就只有一行代碼,唯一的作用就是指向 react 下面的一個子模塊,我們再來看看這個子模塊是怎么寫的:

// react模塊
// react/lib/ReactCSSTransitionGroup.js
var React = require('./React');
var ReactTransitionGroup = require('./ReactTransitionGroup');
var ReactCSSTransitionGroupChild = require('./ReactCSSTransitionGroupChild');
//....剩余代碼忽略

這個子模塊又反回去依賴了 react 整個庫的入口,這就是拖累 Webpack 的罪魁禍首。

總而言之,問題是這樣產(chǎn)生的:

  1. Webpack 發(fā)現(xiàn)我們依賴了 react-addons-css-transition-group
  2. Webpack 去打包 react-addons-css-transition-group 的時候發(fā)現(xiàn)它依賴了 react 模塊下的一個叫 ReactTransitionGroup.js 的文件,于是 Webpack 去打包這個文件。
  3. ReactTransitionGroup.js 依賴了整個 react 的入口文件 React.js,雖然我們設(shè)置了 externals ,但是 Webpack 不知道這個入口文件等效于 react 模塊本身,于是我們可愛又敬業(yè)的 Webpack 就把整個 react 又重新打包了一遍。

讀到這里你可能會有疑問,為什么不能把這個動畫庫也設(shè)置到 externals 里,這樣不是就不用打包了嗎?

問題就在于,這個動畫庫并沒有提供生產(chǎn)環(huán)境的文件,或者說這個庫根本沒有提供 react-addons-css-transition-group.min.js 這個文件。

這個問題不只存在于 react-addons-css-transition-group 中,對于 react 的大多數(shù)現(xiàn)有庫來說都有這個依賴關(guān)系復雜的問題。

初級解決方法

所以對于這個問題的解決方法就是,手工打包這些 module,然后設(shè)置 externals ,讓 Webpack 不再打包它們。

我們需要這樣一個 lib-bundle.js 文件:

window.__LIB["react"] = require("react");
window.__LIB["react-addons-css-transition-group"] = require("react-addons-css-transition-group");
// ...其它依賴包

我們在這里把一些第三方庫注冊到了 window.__LIB 下,這些庫可以作為底層的基礎(chǔ)庫,免于重復打包。

然后執(zhí)行 webpack lib-bundle.js lib.js,得到打包好的 lib.js。然后去設(shè)置我們的 externals :

var webpack = require('webpack');
module.exports = {
 externals: {
  'react': 'window.__LIB["react"]',
  'react-addons-css-transition-group': 'window.__LIB["react-addons-css-transition-group"]',
  // 其它庫
 }
 //其它配置忽略...... 
};

這時由于 externals 的存在,Webpack 打包的時候就會避開這些模塊超多,依賴關(guān)系復雜的庫,把這些第三方 module 的入口指向預先打包好的 lib.js 的入口 window.__LIB,從而只打包我們的業(yè)務代碼。

終極解決方法

上面我們提到的方法本質(zhì)上就是一種動態(tài)鏈接庫(dll)”的思想,這在 windows 系統(tǒng)下面是一種很常見的思想。一個dll包,就是一個很純凈的依賴庫,它本身不能運行,是用來給你的 app 或者業(yè)務代碼引用的。

同樣的 Webpack 最近也新加入了這個功能:webpack.DllPlugin。使用這個功能需要把打包過程分成兩步:

  1. 打包ddl包
  2. 引用ddl包,打包業(yè)務代碼

首先我們來打包ddl包,首先配置一個這樣的 ddl.config.js:

const webpack = require('webpack');

const vendors = [
 'react',
 'react-dom',
 'react-router',
 // ...其它庫
];

module.exports = {
 output: {
  path: 'build',
  filename: '[name].js',
  library: '[name]',
 },
 entry: {
  "lib": vendors,
 },
 plugins: [
  new webpack.DllPlugin({
   path: 'manifest.json',
   name: '[name]',
   context: __dirname,
  }),
 ],
};

webpack.DllPlugin 的選項中:

  1. path 是 manifest.json 文件的輸出路徑,這個文件會用于后續(xù)的業(yè)務代碼打包;
  2. name 是dll暴露的對象名,要跟 output.library 保持一致;
  3. context 是解析包路徑的上下文,這個要跟接下來配置的 webpack.config.js 一致。

運行Webpack,會輸出兩個文件一個是打包好的 lib.js,一個就是 manifest.json,它里面的內(nèi)容大概是這樣的:

{
 "name": "vendor_ac51ba426d4f259b8b18",
 "content": {
  "./node_modules/react/react.js": 1,
  "./node_modules/react/lib/React.js": 2,
  "./node_modules/react/node_modules/object-assign/index.js": 3,
  "./node_modules/react/lib/ReactChildren.js": 4,
  "./node_modules/react/lib/PooledClass.js": 5,
  "./node_modules/react/lib/reactProdInvariant.js": 6,
  // ............
 }
}

接下來我們就可以快樂地打包業(yè)務代碼啦,首先寫好打包配置文件 webpack.config.js:

const webpack = require('webpack');
module.exports = {
 output: {
  path: 'build',
  filename: '[name].js',
 },
 entry: {
  app: './src/index.js',
 },
 plugins: [
  new webpack.DllReferencePlugin({
   context: __dirname,
   manifest: require('./manifest.json'),
  }),
 ],
};

webpack.DllReferencePlugin 的選項中:

  1. context 需要跟之前保持一致,這個用來指導 Webpack 匹配 manifest 中庫的路徑;
  2. manifest 用來引入剛才輸出的 manifest.json 文件。

DllPlugin 本質(zhì)上的做法和我們手動分離這些第三方庫是一樣的,但是對于包極多的應用來說,自動化明顯加快了生產(chǎn)效率。

總結(jié)

以上就是關(guān)于徹底解決Webpack打包慢問題的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

相關(guān)文章

  • showModalDialog 和 showModelessDialog

    showModalDialog 和 showModelessDialog

    showModalDialog 和 showModelessDialog...
    2007-01-01
  • js+css在交互上的應用

    js+css在交互上的應用

    關(guān)于css應用。以前一直認為css就是做布局樣式,只能表現(xiàn)頁面,跟交互是沒關(guān)系的。事實上也基本不會往那邊想。
    2010-07-07
  • javascript結(jié)合Flexbox簡單實現(xiàn)滑動拼圖游戲

    javascript結(jié)合Flexbox簡單實現(xiàn)滑動拼圖游戲

    本文給大家分享的是一則使用javascript結(jié)合Flexbox簡單實現(xiàn)滑動拼圖游戲的代碼,雖然沒有實現(xiàn)完整的功能,但是還是推薦給大家,喜歡的朋友可以繼續(xù)做完
    2016-02-02
  • 使用JavaScript獲取掃碼槍掃描得到的條形碼的思路代碼詳解

    使用JavaScript獲取掃碼槍掃描得到的條形碼的思路代碼詳解

    這篇文章主要介紹了使用JavaScript獲取掃碼槍掃描得到的條形碼的思路代碼詳解,本文通過示例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-06-06
  • TypeScript中interface和type的區(qū)別詳解

    TypeScript中interface和type的區(qū)別詳解

    本文主要介紹了TypeScript中interface和type的區(qū)別詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-07-07
  • js 一個關(guān)于圖片onload加載的事

    js 一個關(guān)于圖片onload加載的事

    前幾天一個項目讓我頭疼了很久,一個關(guān)于圖片加載時的loading效果,因為不是太懂js,所以在網(wǎng)上各種找資料,但還是不理想,無賴苦心研究,終于有了一點眉目了,雖然個中還有一些道理不懂,至少目的達到了
    2013-11-11
  • uniapp電商小程序?qū)崿F(xiàn)訂單30分鐘倒計時

    uniapp電商小程序?qū)崿F(xiàn)訂單30分鐘倒計時

    這篇文章主要為大家詳細介紹了uniapp電商小程序?qū)崿F(xiàn)訂單30分鐘倒計時,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-11-11
  • JS自定義滾動條效果

    JS自定義滾動條效果

    這篇文章主要為大家詳細介紹了JS自定義滾動條效果,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-03-03
  • 微信小程序云開發(fā) 生成帶參小程序碼流程

    微信小程序云開發(fā) 生成帶參小程序碼流程

    這篇文章主要為大家詳細介紹了微信小程序云開發(fā),生成帶參小程序碼,云函數(shù)網(wǎng)絡(luò)請求,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-05-05
  • Javascript 變量作用域 兩個可能會被忽略的小特性

    Javascript 變量作用域 兩個可能會被忽略的小特性

    關(guān)于Javascript,大家肯定都很熟悉啦,對于有編程經(jīng)驗的朋友來說,Javascript很快就能上手,不過關(guān)于JS的變量作用域,還是有一點差別的。
    2010-03-03

最新評論