webpack2 终极优化
ChaPederson
8年前
<p><img src="https://simg.open-open.com/show/a9d59025b1fc5e917931e9d866870c5d.png"></p> <p>webpack是当下最流行的js打包工具,这得益于网页应用日益复杂和js模块化的流行。 <a href="/misc/goto?guid=4959732858242557553" rel="nofollow,noindex">webpack2</a> 增加了一些新特性也到了预发布阶段,是时候告诉大家如何用webpack2优化你的构建让它构建出更小的文件尺寸和更好的开发体验。</p> <h2>优化输出</h2> <p>打包结果更小可以让网页打开速度更快以及简约宽带。可以通过这以下几点做到</p> <p>压缩css</p> <p>css-loader 在webpack2里默认是没有开启压缩的,最后生成的css文件里有很多空格和tab,通过配置 css-loader?minimize 参数可以开启压缩输出最小的css。css的压缩实际是是通过 <a href="/misc/goto?guid=4959732858365234222" rel="nofollow,noindex">cssnano</a> 实现的。</p> <p>tree-shaking</p> <p>tree-shaking 是指借助es6 import export 语法静态性的特点来删掉export但是没有import过的东西。要让tree-shaking工作需要注意以下几点:</p> <ul> <li>配置babel让它在编译转化es6代码时不把 import export 转换为cmd的 module.export ,配置如下: <pre> <code class="language-javascript">"presets": [ [ "es2015", { "modules": false } ] ]</code></pre> </li> <li>大多数分布到npm的库里的代码都是es5的,但是也有部分库(redux,react-router等等)开始支持tree-shaking。这些库发布到npm里的代码即包含es5的又包含全采用了es6 import export 语法的代码。 拿redux库来说,npm下载到的目录结构如下: <pre> <code class="language-javascript">├── es │ └── utils ├── lib │ └── utils</code></pre> 其中lib目录里是编译出的es5代码,es目录里是编译出的采用 import export 语法的es5代码,在redux的 package.json 文件里有这两个配置: <pre> <code class="language-javascript">"main": "lib/index.js", "jsnext:main": "es/index.js",</code></pre> 这是指这个库的入口文件的位置,所以要让webpack去读取es目录下的代码需要使用jsnext:main字段配置的入口,要做到这点webpack需要这样配置: <pre> <code class="language-javascript">module.exports = { resolve: { mainFields: ['jsnext:main','main'], } };</code></pre> 这会让webpack先使用jsnext:main字段,在没有时使用main字段。这样就可以优化支持tree-shaking的库。</li> </ul> <p>优化 UglifyJsPlugin</p> <p>webpack --optimize-minimize 选项会开启 UglifyJsPlugin来压缩输出的js,但是默认的UglifyJsPlugin配置并没有把代码压缩到最小输出的js里还是有注释和空格,需要覆盖默认的配置:</p> <pre> <code class="language-javascript">new UglifyJsPlugin({ // 最紧凑的输出 beautify: false, // 删除所有的注释 comments: false, compress: { // 在UglifyJs删除没有用到的代码时不输出警告 warnings: false, // 删除所有的 `console` 语句 // 还可以兼容ie浏览器 drop_console: true, // 内嵌定义了但是只用到一次的变量 collapse_vars: true, // 提取出出现多次但是没有定义成变量去引用的静态值 reduce_vars: true, } })</code></pre> <p>定义环境变量 NODE_ENV=production</p> <p>很多库里(比如react)有部分代码是这样的:</p> <pre> <code class="language-javascript">if(process.env.NODE_ENV !== 'production'){ // 不是生产环境才需要用到的代码,比如控制台里看到的警告 }</code></pre> <p>在环境变量 NODE_ENV 等于 production 的时候UglifyJs会认为if语句里的是死代码在压缩代码时删掉。</p> <p>DedupePlugin 和 OccurrenceOrderPlugin</p> <p>在webpack1里经常会使用 DedupePlugin 插件来消除重复的模块以及使用 OccurrenceOrderPlugin 插件让被依赖次数更高的模块靠前分到更小的id 来达到输出更少的代码,在webpack2里这些已经这两个插件已经被移除了因为这些功能已经被内置了。</p> <p>除了压缩文本代码外还可以:</p> <ul> <li>用 <a href="/misc/goto?guid=4959732858478477121" rel="nofollow,noindex">imagemin-webpack-plugin</a> 压缩图片</li> <li>用 <a href="/misc/goto?guid=4959732858587152309" rel="nofollow,noindex">webpack-spritesmith</a> 合并雪碧图</li> </ul> <p>以上优化点只需要在构建用于生产环境代码的时候才使用,在开发环境时最好关闭因为它们很耗时。</p> <h2>优化开发体验</h2> <p>优化开发体验主要从更快的构建和更方便的功能入手。</p> <h2>更快的构建</h2> <p>缩小文件搜索范围</p> <p>webpack的 resolve.modules 配置模块库(通常是指node_modules)所在的位置,在js里出现 import 'redux' 这样不是相对也不是绝对路径的写法时会去node_modules目录下找。但是默认的配置会采用向上递归搜索的方式去寻找node_modules,但通常项目目录里只有一个node_modules在项目根目录,为了减少搜索我们直接写明node_modules的全路径:</p> <pre> <code class="language-javascript">module.exports = { resolve: { modules: [path.resolve(__dirname, 'node_modules')] } };</code></pre> <p>除此之外webpack配置loader时也可以缩小文件搜索范围。</p> <ul> <li>loader的test正则表达式也应该尽可能的简单,比如在你的项目里只有 .js 文件时就不要把test写成 /\.jsx?$/</li> <li>loader使用include命中只需要处理的文件,比如babel-loader的这两个配置:</li> </ul> <p>只对项目目录下src目录里的代码进行babel编译</p> <pre> <code class="language-javascript">{ test: /\.js$/, loader: 'babel-loader', include: path.resolve(__dirname, 'src') }</code></pre> <p>项目目录下的所有js都会进行babel编译,包括庞大的node_modules下的js</p> <pre> <code class="language-javascript">{ test: /\.js$/, loader: 'babel-loader' }</code></pre> <p>开启 babel-loader 缓存</p> <p>babel编译过程很耗时,好在babel-loader提供缓存编译结果选项,在重启webpack时不需要创新编译而是复用缓存结果减少编译流程。babel-loader缓存机制默认是关闭的,打开的配置如下:</p> <pre> <code class="language-javascript">module.exports = { module: { loaders: [ { test: /\.js$/, loader: 'babel-loader?cacheDirectory', } ] } };</code></pre> <p>使用 alias</p> <p>resolve.alias 配置路径映射。 发布到npm的库大多数都包含两个目录,一个是放着cmd模块化的lib目录,一个是把所有文件合成一个文件的dist目录,多数的入口文件是指向lib里面下的。 默认情况下webpack会去读lib目录下的入口文件再去递归加载其它依赖的文件这个过程很耗时,alias配置可以让webpack直接使用dist目录的整体文件减少文件递归解析。配置如下:</p> <pre> <code class="language-javascript">module.exports = { resolve: { alias: { 'moment': 'moment/min/moment.min.js', 'react': 'react/dist/react.js', 'react-dom': 'react-dom/dist/react-dom.js' } } };</code></pre> <p>使用 noParse</p> <p>module.noParse 配置哪些文件可以脱离webpack的解析。 有些库是自成一体不依赖其他库的没有使用模块化的,比如jquey、momentjs、chart.js,要使用它们必须整体全部引入。 webpack是模块化打包工具完全没有必要去解析这些文件的依赖,因为它们都不依赖其它文件体积也很庞大,要忽略它们配置如下:</p> <pre> <code class="language-javascript">module.exports = { module: { noParse: /node_modules\/(jquey|moment|chart\.js)/ } };</code></pre> <h2>更方便的功能</h2> <p>模块热替换</p> <p>模块热替换是指在开发的过程中修改代码后不用刷新页面直接把变化的模块替换到老模块让页面呈现出最新的效果。 webpack-dev-server内置模块热替换,配置起来也很方便,下面以react应用为例,步骤如下:</p> <ul> <li>在启动webpack-dev-server的时候带上 --hot 参数开启模块热替换,在开启 --hot 后针对css的变化是会自动热替换的,但是js涉及到复杂的逻辑还需要进一步配置。</li> <li>配置页面入口文件</li> </ul> <pre> <code class="language-javascript">import App from './app'; function run(){ render(<App/>,document.getElementById('app')); } run(); // 只在开发模式下配置模块热替换 if (process.env.NODE_ENV !== 'production') { module.hot.accept('./app', run); }</code></pre> <p>当./app发生变化或者当./app依赖的文件发生变化时会把./app编译成一个模块去替换老的,替换完毕后重新执行run函数渲染出最新的效果。</p> <p>自动生成html</p> <p>webpack只做了资源打包的工作还缺少把这些加载到html里运行的功能,在庞大的app里手写html去加载这些资源是很繁琐易错的,我们需要自动正确的加载打包出的资源。 webpack原生不支持这个功能于是我做了一个插件 <a href="/misc/goto?guid=4959732858692007792" rel="nofollow,noindex">web-webpack-plugin</a> 具体使用点开链接看 <a href="/misc/goto?guid=4959732858804810695" rel="nofollow,noindex">详细文档</a> ,使用大概如下:</p> <p><a href="/misc/goto?guid=4959732858910058614" rel="nofollow,noindex">demo</a></p> <p>webpack配置</p> <pre> <code class="language-javascript">module.exports = { entry: { A: './a', B: './b', }, plugins: [ new WebPlugin({ // 输出的html文件名称,必填,注意不要重名,重名会覆盖相互文件。 filename: 'index.html', // 该html文件依赖的entry,必须是一个数组。依赖的资源的注入顺序按照数组的顺序。 requires: ['A', 'B'], }), ] };</code></pre> <p>将会输出一个 index.html 文件,这个文件将会自动引入 entry A 和 B 生成的js文件,</p> <p>输出的html:</p> <pre> <code class="language-javascript"><!DOCTYPE html> <html> <head> <meta charset="UTF-8"> </head> <body> <script src="A.js"></script> <script src="B.js"></script> </body> </html></code></pre> <p>输出的目录结构</p> <pre> <code class="language-javascript">├── A.js ├── B.js └── index.html</code></pre> <p>最后附上这篇文章所讲到的 <a href="/misc/goto?guid=4959732859009058617" rel="nofollow,noindex">webpack整体的配置</a> ,分为开发环境的 webpack.config.js 和生产环境的 webpack-dist.config.js</p> <p> </p> <p>来自:http://imweb.io/topic/5868e1abb3ce6d8e3f9f99bb</p> <p> </p>