【转】阮一峰 Webpack 教程
原文地址:https://blog.csdn.net/userkang/article/details/83504048
说明:跟着原文学习做的学习笔记(稍有改动,推荐打开原文看完整版)
写在开头
此Webpack教程是阮老师在Webpack1.x的版本上做的。现在Webpack的版本已经改动较大,建议有基础的同学,就直接上官网看最新的就好了。这个教程可以用来了解下 Webpack的前世今生。
仓库地址: https://github.com/userkang/webpack-demos-cn
开始
这个项目是一些简单的 Webpack 示例集合
这些示例特意用简单明了的方式编写,你将会发现跟着这些例子学习这个强大的工具并非难事。
如何使用
首先,全局安装 Webpack 和 webpack-dev-server
$ npm i -g webpack webpack-dev-server
然后,克隆这个仓库
$ git clone https://github.com/userkang/webpack-demos-cn.git
安装依赖
$ cd webpack-demos
$ npm install
现在,去项目 demo* 目录下开始源码体验之旅吧
$ cd demo01
$ npm run dev
如果上面的命令没有自动打开浏览器,你可能需要自己在浏览器访问http://127.0.0.1:8080
Demo01: Entry file
入口文件是 Webpack 进行读取构建 bundle.js 文件的一个文件
例如, main.js 就是一个入口文件.
// main.js document.write('<h1>Hello World</h1>')
//index.html <html> <body> <script type="text/javascript" src="bundle.js"></script> </body> </html>
Webpack 遵循 webpack.config.js 来构建 bundle.js.
//webpack.config.js module.exports = { entry: './main.js', output: { filename: 'bundle.js' } }
启动服务,访问 http://127.0.0.1:8080 .
$ cd demo01
$ npm run dev
Demo02: Multiple entry files
多个入口文件也是可以的。在多页面应用中,每个页面拥有不同的入口文件,用这个就非常管用了。
// main1.js document.write('<h1>Hello World</h1>') // main2.js document.write('<h2>Hello Webpack</h2>')
//index.html <html> <body> <script src="bundle1.js"></script> <script src="bundle2.js"></script> </body> </html>
//webpack.config.js module.exports = { entry: { bundle1: './main1.js', bundle2: './main2.js' }, output: { filename: '[name].js' } }
Demo03: Babel-loader (source)
Loaders 是一种预处理器,它可以在 Webpack 编译之前把你应用中的静态资源进行转换 。
举个例子, Babel-loader 可以在 Webpack 编译这些 JS 文件之前,先将 JSX/ES6 语法的文件转换成普通 ES5 语法的文件。Webpack 官网可以查看目前支持的 loaders。
main.jsx 是一个 JSX 文件.
//main.jsx const React = require('react') const ReactDOM = require('react-dom') ReactDOM.render(<h1>Hello, world!</h1>, document.querySelector('#wrapper'))
//index.html <html> <body> <div id="wrapper"></div> <script src="bundle.js"></script> </body> </html>
//webpack.config.js module.exports = { entry: './main.jsx', output: { filename: 'bundle.js' }, module: { rules: [{ test: /\.jsx?$/, exclude: /node_modules/, use: { loader: 'babel-loader', options: {presets: ['es2015', 'react']} } }] } }
上边的代码段用到了 babel-loader,它需要 Babel 的预设插件 babel-preset-es2015 and babel-preset-react 来转义 ES6 和 React。
Demo04: CSS-loader
Webpack 允许在 JS 文件里包含 CSS,并通过 CSS-loader来预处理 CSS 文件。
//main.js require('./app.css')
//app.css body { background-color: blue; }
//index.html <html> <head> <script type="text/javascript" src="bundle.js"></script> </head> <body> <h1>Hello World</h1> </body> </html>
//webpack.config.js module.exports = { entry: './main.js', output: { filename: 'bundle.js' }, module: { rules: [{ test: /\.css$/, use: ['style-loader', 'css-loader'] }] } }
注意,你必须用到两个loaders 去转换 CSS 文件。一个是 CSS-loader 来读取 CSS 文件,另一个是 Style-loader用来把 <style> 标签插入 HTML 页面。
接下来,让我们启动服务。
$ cd demo04
$ npm run dev
事实上,Webpack 会将一个内联样式表插入到 index.html。
<head> <script type="text/javascript" src="bundle.js"></script> <style type="text/css"> body { background-color: blue; } </style> </head>
Demo05: Image loader
Webpack 可以在 JS 文件中包含图片。
//main.js var img1 = document.createElement('img') img1.src = require('./small.png') document.body.appendChild(img1) var img2 = document.createElement('img') img2.src = require('./big.png') document.body.appendChild(img2)
//index.html <html> <body> <script type="text/javascript" src="bundle.js"></script> </body> </html>
//webpack.config.js module.exports = { entry: './main.js', output: { filename: 'bundle.js' }, module: { rules: [ { test: /\.(png|jpg)$/, use: [ { loader: 'url-loader', options: { limit: 8192 } } ] } ] } }
url-loader将图片文件转成 img 标签。如果图片的大小小于 8192 字节,将会被转成 base64 的格式,否则还事会转成一条普通的链接。
启动服务就可以看到,small.png和big.png这两张图片转成了不同的两种格式。
<img src="...uQmCC">
<img src="4853ca667a2b8b8844eb2693ac1b2578.png">
Demo06: CSS Module (source)
css-loader?modules 的 CSS Module 功能可以给 JS 模块的 CSS 设置一个局部作用域。你可以用 :global(selector) (更多) 关掉它,使样式变成全局的。
//index.html <html> <body> <h1 class="h1">Hello World</h1> <h2 class="h2">Hello Webpack</h2> <div id="example"></div> <script src="./bundle.js"></script> </body> </html>
app.css /* local scope */ .h1 { color: red; } /* global scope */ :global(.h2) { color: blue; }
//main.jsx var React = require('react') var ReactDOM = require('react-dom') var style = require('./app.css') ReactDOM.render( <div> <h1 className={style.h1}>Hello World</h1> <h2 className="h2">Hello Webpack</h2> </div>, document.getElementById('example') )
//webpack.config.js module.exports = { entry: './main.jsx', output: { filename: 'bundle.js' }, module: { rules: [ { test: /\.js[x]?$/, exclude: /node_modules/, use: { loader: 'babel-loader', options: { presets: ['es2015', 'react'] } } }, { test: /\.css$/, use: [ { loader: 'style-loader' }, { loader: 'css-loader', // 开启 CSS Module 功能 options: { modules: true } } ] } ] } }
启动服务。访问 http://127.0.0.1:8080, 你将看到 h1 是红色的,因为它的样式是局部作用域,h2 是蓝色的,因为它的作用域是全局的。
Demo07: UglifyJs Plugin
Webpack 用一套插件系统扩展了它的功能。比如,UglifyJs Plugin 就是其中的一个流行的插件,它可以压缩混淆输出的 js 代码。
//main.js var longVariableName = 'Hello' longVariableName += ' World' document.write('<h1>' + longVariableName + '</h1>')
//index.html <html> <body> <script src="bundle.js"></script> </body> </html>
//webpack.config.js var webpack = require('webpack') var UglifyJsPlugin = require('uglifyjs-webpack-plugin') module.exports = { entry: './main.js', output: { filename: 'bundle.js' }, // 使用插件 plugins: [new UglifyJsPlugin()] }
启动服务,将会看到 main.js 被压缩成了下面的样式
var o = 'Hello' ;(o += ' World'), document.write('<h1>' + o + '</h1>')
Demo08: HTML Webpack Plugin and Open Browser Webpack Plugin
这个例子将讲讲怎么载入第三方的插件。
html-webpack-plugin 将会为你创建一个 index.html 文件。
open-browser-webpack-plugin 可以在 Webpack 编译完成后打开一个新窗口。
//main.js document.write('<h1>Hello World</h1>')
//webpack.config.js var HtmlwebpackPlugin = require('html-webpack-plugin') var OpenBrowserPlugin = require('open-browser-webpack-plugin') module.exports = { entry: './main.js', output: { filename: 'bundle.js' }, plugins: [ new HtmlwebpackPlugin({ title: 'Webpack-demos', filename: 'index.html' }), new OpenBrowserPlugin({ url: 'http://localhost:8080' }) ] }
启动服务。现在你不仅不需要手动的写 index.html 文件,而且也不用手动的打开浏览器了。Webpack 都帮你完成了。
Demo09: Environment flags
你可以用环境变量来区分开发环境。
//main.js document.write('<h1>Hello World</h1>') if (__DEV__) { document.write(new Date()) }
//index.html <html> <body> <script src="bundle.js"></script> </body> </html>
//webpack.config.js var webpack = require('webpack')
// DefinePlugin 允许创建一个在编译时可以配置的全局常量 var devFlagPlugin = new webpack.DefinePlugin({ __DEV__: JSON.stringify(JSON.parse(process.env.DEBUG || 'false')) }) module.exports = { entry: './main.js', output: { filename: 'bundle.js' }, plugins: [devFlagPlugin] }
现在打开 demo09/package.json,你需要在 scripts 字段下添加如下的代码,向 webpack 传递环境变量。
// package.json { // ... "scripts": { "dev": "cross-env DEBUG=true webpack-dev-server --open", }, // ... }
启动服务。
Demo10: Code splitting
对于一个大型 app,把所有代码塞到一个文件可能不是很好维护。Webpack 可以把一个庞大的 JS 文件拆分成几块。尤其,有些代码只有用的时候才引入可以按需加载。Webpack 用 require.ensure 来定义一个代码点。
// main.js require.ensure(['./a'], function(require) { var content = require('./a') document.open() document.write('<h1>' + content + '</h1>') document.close() })
require.ensure 告诉 Webpack ./a.js 应该从 bundle.js 中拆分出来,单独生成一个文件。
// a.js module.exports = 'Hello World'
现在 Webpack 来负责依赖、输出和运行时需要的一些东西。你不必在投入过多的精力在 index.html 和 webpack.config.js 上了。
//index.html <html> <body> <script src="bundle.js"></script> </body> </html> //webpack.config.js module.exports = { entry: './main.js', output: { filename: 'bundle.js' } }
启动服务。表面上,你不会感到什么变化。然而,Webpack 已经把 main.js 和 a.js 构建成两个不同的文件(bundle.js and 0.bundle.js),并且按依赖关系依次引入。
Demo11: Code splitting with bundle-loader
另一种进行代码分离的方法是用 bundle-loader。
// main.js // 现在 a.js 被引入,它将会被打包到另一个文件 var load = require('bundle-loader!./a.js') // 为了等待 a.js 中的模块变得可用 // 你可能需要使用 async wait load(function(file) { document.open() document.write('<h1>' + file + '</h1>') document.close() })
require('bundle-loader!./a.js') 告诉 Webpack 从另一个包里加载 a.js。
现在 Webpack 将会构建 main.js 到 bundle.js 中, a.js 到 0.bundle.js 中。
Demo12: Common chunk
当多个 Js 文件有共同的依赖,我们可以 CommonsChunkPlugin 把公共的部分提取出来生成一个文件。这对浏览器缓存和节省带宽是非常有用的。
// main1.jsx var React = require('react') var ReactDOM = require('react-dom') ReactDOM.render(<h1>Hello World</h1>, document.getElementById('a')) // main2.jsx var React = require('react') var ReactDOM = require('react-dom') ReactDOM.render(<h2>Hello Webpack</h2>, document.getElementById('b'))
index.html <html> <body> <div id="a"></div> <div id="b"></div> <script src="commons.js"></script> <script src="bundle1.js"></script> <script src="bundle2.js"></script> </body> </html>
上面的 commons.js 就是 main1.jsx 和 main2.jsx 的公共部分。也就是包含了 react 和 react-dom。
webpack.config.js var webpack = require('webpack') module.exports = { entry: { bundle1: './main1.jsx', bundle2: './main2.jsx' }, output: { filename: '[name].js' }, module: { rules: [ { test: /\.js[x]?$/, exclude: /node_modules/, use: { loader: 'babel-loader', options: { presets: ['es2015', 'react'] } } } ] }, plugins: [ new webpack.optimize.CommonsChunkPlugin({ name: 'commons', // (the commons chunk name) filename: 'commons.js' // (the filename of the commons chunk) }) ] }
Demo12: Common chunk
当多个 Js 文件有共同的依赖,我们可以 CommonsChunkPlugin 把公共的部分提取出来生成一个文件。这对浏览器缓存和节省带宽是非常有用的。
// main1.jsx var React = require('react') var ReactDOM = require('react-dom') ReactDOM.render(<h1>Hello World</h1>, document.getElementById('a')) // main2.jsx var React = require('react') var ReactDOM = require('react-dom') ReactDOM.render(<h2>Hello Webpack</h2>, document.getElementById('b'))
//index.html <html> <body> <div id="a"></div> <div id="b"></div> <script src="commons.js"></script> <script src="bundle1.js"></script> <script src="bundle2.js"></script> </body> </html>
上面的 commons.js 就是 main1.jsx 和 main2.jsx 的公共部分。也就是包含了 react 和 react-dom。
Demo13: Vendor chunk
你也可以用 CommonsChunkPlugin 提取第三方的 Js 生成一个单独的文件。
//main.js var $ = require('jquery') $('h1').text('Hello World') //index.html <html> <body> <h1></h1> <script src="vendor.js"></script> <script src="bundle.js"></script> </body> </html>
//webpack.config.js
var webpack = require('webpack')
module.exports = {
entry: {
app: './main.js',
vendor: ['jquery']
},
output: {
filename: 'bundle.js'
},
plugins: [
new webpack.optimize.CommonsChunkPlugin({
name: 'vendor',
filename: 'vendor.js'
})
]
}
上面的代码,entry.vendor: ['jquery'] 告诉 Webpack jeuery 应该被打包到一个公共 vendor.js 包中去。
如果你想让一个全局的变量在每一个模块可用,比如 $ 和 jQuery 不用 require("jquery") 就可以直接用。那么你需要使用 ProvidePlugin (Official doc) 这个插件,它可以自动的载入模块,而不需要到处 import 或者 require。
// main.js $('h1').text('Hello World') // webpack.config.js var webpack = require('webpack') module.exports = { entry: { app: './main.js' }, output: { filename: 'bundle.js' }, plugins: [ new webpack.ProvidePlugin({ $: 'jquery', jQuery: 'jquery' }) ] }
当然,这种情况,你还需要手动的全局的载入 jquery.js
Demo14: Exposing global variables
如果你想要用到一些全局变量,但是又不想在打包的时候,把它打包到 bundle 文件中去。这个时候你可以在 webpack.config.js 中配置 externals 字段。
比如,我们定义了一个 data.js 文件
// data.js var data = 'Hello World' //index.html <html> <body> <script src="data.js"></script> <script src="bundle.js"></script> </body> </html>
注意,这里 Webpack 只会打包 bundle.js, 而不会打包 data.js。
我们可以 data 字段暴露盛一个全局变量
//webpack.config.js module.exports = { entry: './main.jsx', output: { filename: 'bundle.js' }, module: { rules: [ { test: /\.js[x]?$/, exclude: /node_modules/, use: { loader: 'babel-loader', options: { presets: ['es2015', 'react'] } } } ] }, externals: { // require('data') is external and available // on the global var data data: 'data' } }
现在,你可以在脚本中像引入其他模块一样引入 data。但它实际是一个全局变量。
// main.jsx var data = require('data') var React = require('react') var ReactDOM = require('react-dom') ReactDOM.render(<h1>{data}</h1>, document.body)
你同样可以把 react 和 react-dom 添加到 externals 中,这样可以大大缩短编译时间和编译后 bundle.js 文件的大小。