webpack原理与实战

webpack是一个js打包工具,不一个完整的前端构建工具。它的流行得益于模块化和单页应用的流行。webpack提供扩展机制,在庞大的社区支持下各类场景基本它均可找到解决方案。本文的目的是教会你用webpack解决实战中常见的问题。css

webpack原理

在深刻实战前先要知道webpack的运行原理html

webpack核心概念

  • entry 一个可执行模块或库的入口文件。
  • chunk 多个文件组成的一个代码块,例如把一个可执行模块和它全部依赖的模块组合和一个 chunk 这体现了webpack的打包机制。
  • loader 文件转换器,例如把es6转换为es5,scss转换为css。
  • plugin 插件,用于扩展webpack的功能,在webpack构建生命周期的节点上加入扩展hook为webpack加入功能。

webpack构建流程

从启动webpack构建到输出结果经历了一系列过程,它们是:前端

  1. 解析webpack配置参数,合并从shell传入和webpack.config.js文件里配置的参数,生产最后的配置结果。
  2. 注册全部配置的插件,好让插件监听webpack构建生命周期的事件节点,以作出对应的反应。
  3. 从配置的entry入口文件开始解析文件构建AST语法树,找出每一个文件所依赖的文件,递归下去。
  4. 在解析文件递归的过程当中根据文件类型和loader配置找出合适的loader用来对文件进行转换。
  5. 递归完后获得每一个文件的最终结果,根据entry配置生成代码块chunk
  6. 输出全部chunk到文件系统。

须要注意的是,在构建生命周期中有一系列插件在合适的时机作了合适的事情,好比UglifyJsPlugin会在loader转换递归完后对结果再使用UglifyJs压缩覆盖以前的结果。node

场景和方案

经过各类场景和对应的解决方案让你深刻掌握webpackreact

单页应用

demo redemo
一个单页应用须要配置一个entry指明执行入口,webpack会为entry生成一个包含这个入口全部依赖文件的chunk,但要让它在浏览器里跑起来还须要一个HTML文件来加载chunk生成的js文件,若是提取出了css还须要让HTML文件引入提取出的css。web-webpack-plugin里的WebPlugin能够自动的完成这些工做。webpack

webpack配置文件git

const { WebPlugin } = require('web-webpack-plugin'); module.exports = { entry: { app: './src/doc/index.js', }, plugins: [ // 一个WebPlugin对应生成一个html文件 new WebPlugin({ //输出的html文件名称 filename: 'index.html', //这个html依赖的`entry` requires: ['app'], }), ], };

requires: ['doc']指明这个HTML依赖哪些entryentry生成的js和css会自动注入到HTML里。
你还能够配置这些资源的注入方式,支持以下属性:es6

  • _dist 只有在生产环境下才引入该资源
  • _dev 只有在开发环境下才引入该资源
  • _inline 把该资源的内容潜入到html里
  • _ie 只有IE浏览器才须要引入的资源

要设置这些属性能够经过在js里配置github

new WebPlugin({ filename: 'index.html', requires: { app:{ _dist:true, _inline:false, } }, }),

或者在模版里设置,使用模版的好处是灵活的控制资源注入点。web

new WebPlugin({ filename: 'index.html', template: './template.html', }),
<!DOCTYPE html>
<html lang="zh-cn"> <head> <link rel="stylesheet" href="app?_inline"> <script src="ie-polyfill?_ie"></script> </head> <body> <div id="react-body"></div> <script src="app"></script> </body> </html>

WebPlugin插件借鉴了fis3的思想,补足了webpack缺失的以HTML为入口的功能。想了解WebPlugin的更多功能,见文档

一个项目里管理多个单页应用

通常项目里会包含多个单页应用,虽然多个单页应用也能够合并成一个可是这样作会致使用户没访问的部分也加载了。若是项目里有不少个单页应用,为每一个单页应用配置一个entryWebPlugin?若是项目又新增了一个单页应用,又去新增webpack配置?这样作太麻烦了,web-webpack-plugin里的AutoWebPlugin能够方便的解决这些问题。

module.exports = { plugins: [ // 全部页面的入口目录 new AutoWebPlugin('./src/'), ] };

AutoWebPlugin会把./src/目录下全部每一个文件夹做为一个单页页面的入口,自动为全部的页面入口配置一个WebPlugin输出对应的html。要新增一个页面就在./src/下新建一个文件夹包含这个单页应用所依赖的代码,AutoWebPlugin自动生成一个名叫文件夹名称的html文件。AutoWebPlugin的更多功能见文档

代码分割优化

一个好的代码分割对浏览器首屏效果提高很大。好比对于最多见的react体系你能够

  1. 先抽出基础库react react-dom redux react-redux到一个单独的文件而不是和其它文件放在一块儿打包为一个文件,这样作的好处是只要你不升级他们的版本这个文件永远不会被刷新。若是你把这些基础库和业务代码打包在一个文件里每次改动业务代码都会致使文件hash值变化从而致使缓存失效浏览器重复下载这些包含基础库的代码。以上的配置为:
// vender.js 文件抽离基础库到单独的一个文件里防止跟随业务代码被刷新 // 全部页面都依赖的第三方库 // react基础 import 'react'; import 'react-dom'; import 'react-redux'; // redux基础 import 'redux'; import 'redux-thunk';
// webpack配置 { entry: { vendor: './path/to/vendor.js', }, }
  1. 再经过CommonsChunkPlugin能够提取出多个代码块都依赖的代码造成一个单独的chunk。在应用有多个页面的场景下提取出全部页面公共的代码减小单个页面的代码,在不一样页面之间切换时全部页面公共的代码以前被加载过而没必要从新加载。

构建npm包

demo remd
除了构建可运行的web应用,webpack也可用来构建发布到npm上去的给别人调用的js库。

const nodeExternals = require('webpack-node-externals'); module.exports = { entry: { index: './src/index.js', }, externals: [nodeExternals()], target: 'node', output: { path: path.resolve(__dirname, '.npm'), filename: '[name].js', libraryTarget: 'commonjs2', }, };

这里有几个区别于web应用不一样的地方:

  • externals: [nodeExternals()]用于排除node_modules目录下的代码被打包进去,由于放在node_modules目录下的代码应该经过npm安装。
  • libraryTarget: 'commonjs2'指出entry是一个可供别人调用的库而不是可执行的,输出的js文件按照commonjs规范。

构建服务端渲染

服务端渲染的代码要运行在nodejs环境,和浏览器不一样的是,服务端渲染代码须要采用commonjs规范同时不该该包含除js以外的文件好比css。webpack配置以下:

module.exports = { target: 'node', entry: { 'server_render': './src/server_render', }, output: { filename: './dist/server/[name].js', libraryTarget: 'commonjs2', }, module: { rules: [ { test: /\.js$/, loader: 'babel-loader', }, { test: /\.(scss|css|pdf)$/, loader: 'ignore-loader', }, ] }, };

其中几个关键的地方在于:

  • target: 'node' 指明构建出的代码是要运行在node环境里
  • libraryTarget: 'commonjs2' 指明输出的代码要是commonjs规范
  • {test: /\.(scss|css|pdf)$/,loader: 'ignore-loader'} 是为了防止不能在node里执行服务端渲染也用不上的文件被打包进去。

从fis3迁移到webpack

fis3和webpack有类似的地方也有不一样的地方。类似在于他们都采用commonjs规范,不一样在于导入css这些非js资源的方式。fis3经过// @require './index.scss'而webpack经过require('./index.scss')。若是想从fis3平滑迁移到webpack可使用comment-require-loader。好比你想在webpack构建是使用采用了fis3方式的imui模块,配置以下:

loaders:[{
     test: /\.js$/, loaders: ['comment-require-loader'], include: [path.resolve(__dirname, 'node_modules/imui'),] }]

自定义webpack扩展

若是你在社区找不到你的应用场景的解决方案,那就须要本身动手了写loader或者plugin了。
在你编写自定义webpack扩展前你须要想明白究竟是要作一个loader仍是plugin呢?能够这样判断:

若是你的扩展是想对一个个单独的文件进行转换那么就编写loader剩下的都是plugin

其中对文件进行转换能够是像:

  • babel-loader把es6转换成es5
  • file-loader把文件替换成对应的URL
  • raw-loader注入文本文件内容到代码里去

编写 webpack loader

demo comment-require-loader
编写loader很是简单,以comment-require-loader为例:

module.exports = function (content) { return replace(content); };

loader的入口须要导出一个函数,这个函数要干的事情就是转换一个文件的内容。
函数接收的参数content是一个文件在转换前的字符串形式内容,须要返回一个新的字符串形式内容做为转换后的结果,全部经过模块化倒入的文件都会通过loader。从这里能够看出loader只能处理一个个单独的文件而不能处理代码块。想编写更复杂的loader可参考官方文档

编写 webpack plugin

demo end-webpack-plugin
plugin应用场景普遍,因此稍微复杂点。以end-webpack-plugin为例:

class EndWebpackPlugin { constructor(doneCallback, failCallback) { this.doneCallback = doneCallback; this.failCallback = failCallback; } apply(compiler) { // 监听webpack生命周期里的事件,作相应的处理 compiler.plugin('done', (stats) => { this.doneCallback(stats); }); compiler.plugin('failed', (err) => { this.failCallback(err); }); } } module.exports = EndWebpackPlugin;

loader的入口须要导出一个class, 在new EndWebpackPlugin()的时候经过构造函数传入这个插件须要的参数,在webpack启动的时候会先实例化plugin再调用pluginapply方法,插件须要在apply函数里监听webpack生命周期里的事件,作相应的处理。
webpack plugin 里有2个核心概念:

  • Compiler: 从webpack启动到推出只存在一个CompilerCompiler存放着webpack配置
  • Compilation: 因为webpack的监听文件变化自动编译机制,Compilation表明一次编译。

Compiler 和 Compilation 都会广播一系列事件。
webpack生命周期里有很是多的事件能够在event-hooksCompilation里查到。以上只是一个最简单的demo,更复杂的能够查看 how to write a plugin或参考web-webpack-plugin

总结

webpack其实很简单,能够用一句话涵盖它的本质:

webpack是一个打包模块化js的工具,能够经过loader转换文件,经过plugin扩展功能。

若是webpack让你感到复杂,必定是各类loader和plugin的缘由。但愿本文能让你明白webpack的原理与本质让你能够在实战中灵活应用webpack。

相关文章
相关标签/搜索