webpack高手秘籍(一)
前言
前面写了一篇文章webpack源码解析一梳理了一遍webpack的编译过程,今天我们结合demo来过一遍webpack的所有配置项。
开始
为了更好的了解每一个选项的用法我们首先clone一份webpack的源码(demo用的是5.0.0-beta.7版本):
git clone https://github.com/webpack/webpack.git
源码clone完毕后在当前页面执行setup操作并且安装webpack-cli:
yarn setup && yarn add webpack-cli -D
ok!接着我们创建一个webpack-demo目录(demo代码上传到github了,也可以直接clone),在当前目录执行npm初始化:
mkdir webpack-demo && npm init
初始化完毕后我们在根目录创建一个webpack的配置文件webpack.config.js
,里面默认导出一个空对象,
webpack.config.js:
module.exports = {
}
然后我们在根目录创建一个src目录用来存放源码,最后我们直接在项目根目录去依赖webpack的源码:
npm install -D xxx/webpack/webpack-src
依赖完毕后我们在根目录执行webpack测试一下:
➜ webpack-demo git:(master) ✗ npx webpack
Hash: 2b5ba38fa93a63c41d7d
Version: webpack 5.0.0-beta.7
Time: 58ms
Built at: 2020-07-07 14:11:47
1 asset
Entrypoint main = main.js
WARNING in configuration
The 'mode' option has not been set, webpack will fallback to 'production' for this value. Set 'mode' option to 'development' or 'production' to enable defaults for each environment.
You can also set it to 'none' to disable any default behavior. Learn more: https://webpack.js.org/configuration/mode/
ERROR in main
Module not found: Error: Can't resolve './src' in 'xxx/webpack/webpack-demo'
可以看到,报了一个错误跟一个警告,说我们没有设置mode
选项、src底下没有发现模块信息,ok! 下面我们正式进入到webpack的配置选项。
配置
模式(mode)
string: developme、production()、none
Option | Description |
---|---|
development |
利用 DefinePlugin 插件设置process.env.NODE_ENV 值为 development . 打开 NamedChunksPlugin and NamedModulesPlugin 插件 |
production |
利用 DefinePlugin 插件设置process.env.NODE_ENV 值为 production . 打开 FlagDependencyUsagePlugin , FlagIncludedChunksPlugin , ModuleConcatenationPlugin , NoEmitOnErrorsPlugin , OccurrenceOrderPlugin , SideEffectsFlagPlugin and TerserPlugin 插件。 |
none |
Opts out of any default optimization options |
Mode: development
// webpack.development.config.js
module.exports = {
+ mode: 'development'
- devtool: 'eval',
- cache: true,
- performance: {
- hints: false
- },
- output: {
- pathinfo: true
- },
- optimization: {
- namedModules: true,
- namedChunks: true,
- nodeEnv: 'development',
- flagIncludedChunks: false,
- occurrenceOrder: false,
- concatenateModules: false,
- splitChunks: {
- hidePathInfo: false,
- minSize: 10000,
- maxAsyncRequests: Infinity,
- maxInitialRequests: Infinity,
- },
- noEmitOnErrors: false,
- checkWasmTypes: false,
- minimize: false,
- removeAvailableModules: false
- },
- plugins: [
- new webpack.NamedModulesPlugin(),
- new webpack.NamedChunksPlugin(),
- new webpack.DefinePlugin({ "process.env.NODE_ENV": JSON.stringify("development") }),
- ]
}
Mode: production
// webpack.production.config.js
module.exports = {
+ mode: 'production',
- performance: {
- hints: 'warning'
- },
- output: {
- pathinfo: false
- },
- optimization: {
- namedModules: false,
- namedChunks: false,
- nodeEnv: 'production',
- flagIncludedChunks: true,
- occurrenceOrder: true,
- concatenateModules: true,
- splitChunks: {
- hidePathInfo: true,
- minSize: 30000,
- maxAsyncRequests: 5,
- maxInitialRequests: 3,
- },
- noEmitOnErrors: true,
- checkWasmTypes: true,
- minimize: true,
- },
- plugins: [
- new TerserPlugin(/* ... */),
- new webpack.DefinePlugin({ "process.env.NODE_ENV": JSON.stringify("production") }),
- new webpack.optimize.ModuleConcatenationPlugin(),
- new webpack.NoEmitOnErrorsPlugin()
- ]
}
Mode: none
// webpack.custom.config.js
module.exports = {
+ mode: 'none',
- performance: {
- hints: false
- },
- optimization: {
- flagIncludedChunks: false,
- occurrenceOrder: false,
- concatenateModules: false,
- splitChunks: {
- hidePathInfo: false,
- minSize: 10000,
- maxAsyncRequests: Infinity,
- maxInitialRequests: Infinity,
- },
- noEmitOnErrors: false,
- checkWasmTypes: false,
- minimize: false,
- },
- plugins: []
}
👌,我们暂时先不分析每一个插件的用法,后面会讲到,我们直接把demo的mode设置成development
:
module.exports = {
mode: "development"
};
然后再运行我们会发现警告没有了:
webpack-demo git:(master) ✗ npx webpack
Hash: 52512bca8884d07d6c74
Version: webpack 5.0.0-beta.7
Time: 55ms
Built at: 2020-07-07 14:29:16
Asset Size
main.js 77 bytes [compared for emit] [name: main]
Entrypoint main = main.js
ERROR in main
Module not found: Error: Can't resolve './src' in 'xxx/webpack/webpack-demo'
➜ webpack-demo git:(master) ✗
如果你想根据 mode 变量值来配置 webpack.config.js, 你可以在配置文件中导出方法而不是变量:
const config = {
};
module.exports = (env, argv) => {
console.log(argv.mode)
if (argv.mode === 'development') {
config.devtool = 'source-map';
}
if (argv.mode === 'production') {
//...
}
return config;
};
运行webpack看结果:
➜ webpack-demo git:(master) ✗ npx webpack --mode development
development
Hash: f6a2edaae6eeb526600f
Version: webpack 5.0.0-beta.7
Time: 41ms
Built at: 2020-07-07 14:45:01
Asset Size
main.js 110 bytes [compared for emit] [name: main]
main.js.map 84 bytes [compared for emit] [dev] [name: (main)]
Entrypoint main = main.js (main.js.map)
ERROR in main
Module not found: Error: Can't resolve './src' in 'xxx/webpack/webpack-demo'
➜ webpack-demo git:(master) ✗
源码
xxx/webpack-src/lib/webpack.js:
...
const createCompiler = options => {
options = new WebpackOptionsDefaulter().process(options);
...
compiler.options = new WebpackOptionsApply().process(options, compiler);
return compiler;
};
..
可以看到,options
被 WebpackOptionsDefaulter
和WebpackOptionsApply
处理过后直接给了compiler
,看名字应该可以看懂,就是webpack的默认配置的意思,
lib/WebpackOptionsDefaulter.js:
...
class WebpackOptionsDefaulter extends OptionsDefaulter {
constructor() {
super();
this.set("entry", "./src"); //设置默认入口为./src
this.set("context", process.cwd()); //设置context默认值为当前目录
this.set("target", "web"); //设置target默认值为web
...
this.set("devtool", "make", options =>
options.mode === "development" ? "eval" : false
);
}
}
...
lib/WebpackOptionsApply.js:
...
class WebpackOptionsApply extends OptionsApply {
constructor() {
super();
}
/**
* @param {WebpackOptions} options options object
* @param {Compiler} compiler compiler object
* @returns {WebpackOptions} options object
*/
process(options, compiler) {
compiler.outputPath = options.output.path;
compiler.recordsInputPath = options.recordsInputPath || options.recordsPath;
compiler.recordsOutputPath =
options.recordsOutputPath || options.recordsPath;
compiler.name = options.name;
if (typeof options.target === "string") {
switch (options.target) {
case "web": {
const JsonpTemplatePlugin = require("./web/JsonpTemplatePlugin");
const FetchCompileWasmPlugin = require("./web/FetchCompileWasmPlugin");
const FetchCompileAsyncWasmPlugin = require("./web/FetchCompileAsyncWasmPlugin");
const NodeSourcePlugin = require("./node/NodeSourcePlugin");
new JsonpTemplatePlugin().apply(compiler);
new FetchCompileWasmPlugin({
mangleImports: options.optimization.mangleWasmImports
}).apply(compiler);
new FetchCompileAsyncWasmPlugin().apply(compiler);
new NodeSourcePlugin(options.node).apply(compiler);
new LoaderTargetPlugin(options.target).apply(compiler);
break;
}
...
可以看到,会根据配置信息加载默认的一些插件, 我们现在是在写demo,为了方便看懂源码我们直接把mode
的值设置成development
,
webpack.config.js:
module.exports = {
mode: "development"
};
Entry and Context(入口文件和上下文)
webpack打包的入口文件配置。
context
string
基础目录,绝对路径,用于从配置中解析入口起点(entry point)和 loader
context: path.resolve(__dirname, ".")
默认使用当前目录,但是推荐在配置中传递一个值。这使得你的配置独立于 CWD(current working directory - 当前执行路径)。
我们在上面默认配置的源码中有提到,
ib/WebpackOptionsDefaulter.js:
...
class WebpackOptionsDefaulter extends OptionsDefaulter {
constructor() {
super();
this.set("entry", "./src"); //设置默认入口为./src
this.set("context", process.cwd()); //设置context默认值为当前目录
this.set("target", "web"); //设置target默认值为web
);
}
}
...
建议我们设置一个当前根目录的路径,ok! 我们修改一下我们的配置文件,然后把context
的值设置为当前项目根目录:
webpack.config.js
const path = require("path");
module.exports = {
mode: "development",
context: path.resolve(__dirname, ".")
};
👌,我们设置了一个当前项目根目录给context,然后我们运行一下webpack:
➜ webpack-demo git:(master) ✗ npx webpack --mode development
Hash: 52512bca8884d07d6c74
Version: webpack 5.0.0-beta.7
Time: 39ms
Built at: 2020-07-07 15:31:37
Asset Size
main.js 77 bytes [compared for emit] [name: main]
Entrypoint main = main.js
ERROR in main
Module not found: Error: Can't resolve './src' in 'xxx/webpack/webpack-demo'
➜ webpack-demo git:(master) ✗
我们可以看到,当我们设置了当前目录给context
的时候,webpack会去context目录去加载默认的入口文件"./src",我们把context直接指向src目录试试:
webpack.config.js
const path = require("path");
module.exports = {
mode: "development",
context: path.resolve(__dirname, "./src")
};
运行webpack:
➜ webpack-demo git:(master) ✗ npx webpack --mode development
Hash: 0845667d00620459218b
Version: webpack 5.0.0-beta.7
Time: 43ms
Built at: 2020-07-07 15:34:46
Asset Size
main.js 77 bytes [compared for emit] [name: main]
Entrypoint main = main.js
ERROR in main
Module not found: Error: Can't resolve './src' in 'xxx/webpack/webpack-demo/src'
➜ webpack-demo git:(master) ✗
可以看到,webpack就会去"./src"目录中去找默认的入口文件"./src/src"。
entry
string
[string]
object = { string | [string] | object = { import string | [string], dependOn string | [string], filename string }}
(function() => string | [string] | object = { string | [string] } | object = { import string | [string], dependOn string | [string], filename string })
起点或是应用程序的起点入口。从这个起点开始,应用程序启动执行。如果传递一个数组,那么数组的每一项都会执行。
动态加载的模块不是入口起点。
简单规则:每个 HTML 页面都有一个入口起点。单页应用(SPA):一个入口起点,多页应用(MPA):多个入口起点。
entry: {
home: "./home.js",
about: "./about.js",
contact: "./contact.js"
}
命名
如果传入一个字符串或字符串数组,chunk 会被命名为 main
。如果传入一个对象,则每个键(key)会是 chunk 的名称,该值描述了 chunk 的入口起点。
ok,我们在src下面创建一个index.js文件用来测试,
src/index.js:
console.log("hello webpack");
webpack.config.js:
const path = require("path");
module.exports = {
mode: "development",
context: path.resolve(__dirname, "./src"),
entry: "./index.js"
};
我们直接使用了一个字符串来声明entry
,然后执行webpack:
➜ webpack-demo git:(master) ✗ npx webpack
Hash: 1435b4d06d79acb2f57c
Version: webpack 5.0.0-beta.7
Time: 55ms
Built at: 2020-07-07 15:40:49
Asset Size
main.js 289 bytes [emitted] [name: main]
Entrypoint main = main.js
./index 29 bytes [built]
➜ webpack-demo git:(master) ✗
Ok,这一次没有报错了,然后提示我们生成了一个“main.js"文件,
dist/main.js(webpack 编译过后的文件):
/******/ (() => { // webpackBootstrap
/*!***************!*\
!*** ./index ***!
\***************/
/*! exports [maybe provided (runtime-defined)] [no usage info] */
/*! runtime requirements: */
eval("console.log(\"hello webpack\");\n\n//# sourceURL=webpack:///./index?");
/******/ })()
;
可以看到,如果没有指定entry的key值的话,默认输出就是main.js。
字符串
entry: "./index.js"
这个我们上面已经测试过了。
数组
entry: ['./index.js']
之前我们研究过babel,如果我们需要手动的在入口文件中导入@babel/polyfill的话,可以用以下配置:
const path = require("path");
module.exports = {
mode: "development",
context: path.resolve(__dirname, "./src"),
entry: ["babel-polyfill","./index.js"]
};
我们下一个polyfill试试,然后执行webpack编译看结果:
npm install -S babel-polyfill && npx webpack
dist/main.js:
/******/ (() => { // webpackBootstrap
/******/ var __webpack_modules__ = ({
/***/ "../node_modules/core-js/fn/regexp/escape.js":
/*!***************************************************!*\
!*** ../node_modules/core-js/fn/regexp/escape.js ***!
\***************************************************/
/*! exports [maybe provided (runtime-defined)] [no usage info] */
/*! runtime requirements: __webpack_require__, module */
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
eval("__webpack_require__(/*! ../../modules/core.regexp.escape */ \"../node_modules/core-js/modules/core.regexp.escape.js\");\nmodule.exports = __webpack_require__(/*! ../../modules/_core */ \"../node_modules/core-js/modules/_core.js\").RegExp.escape;\n\n\n//# sourceURL=webpack:///../node_modules/core-js/fn/regexp/escape.js?");
/***/ }),
/***/ "../node_modules/core-js/modules/_a-function.js":
/*!******************************************************!*\
!*** ../node_modules/core-js/modules/_a-function.js ***!
\******************************************************/
...
eval("\n\n__webpack_require__(/*! core-js/shim */ \"../node_modules/core-js/shim.js\");\n\n__webpack_require__(/*! regenerator-runtime/runtime */ \"../node_modules/regenerator-runtime/runtime.js\");\n\n__webpack_require__(/*! core-js/fn/regexp/escape */ \"../node_modules/core-js/fn/regexp/escape.js\");\n\nif (__webpack_require__.g._babelPolyfill) {\n throw new Error(\"only one instance of babel-polyfill is allowed\");\n}\n__webpack_require__.g._babelPolyfill = true;\n\nvar DEFINE_PROPERTY = \"defineProperty\";\nfunction define(O, key, value) {\n O[key] || Object[DEFINE_PROPERTY](O, key, {\n writable: true,\n configurable: true,\n value: value\n });\n}\n\ndefine(String.prototype, \"padLeft\", \"\".padStart);\ndefine(String.prototype, \"padRight\", \"\".padEnd);\n\n\"pop,reverse,shift,keys,values,entries,indexOf,every,some,forEach,map,filter,find,findIndex,includes,join,slice,concat,push,splice,unshift,sort,lastIndexOf,reduce,reduceRight,copyWithin,fill\".split(\",\").forEach(function (key) {\n [][key] && define(Array, key, Function.call.bind([][key]));\n});\n\n//# sourceURL=webpack:///../node_modules/babel-polyfill/lib/index.js?");
}();
!function() {
/*!******************!*\
!*** ./index.js ***!
\******************/
/*! exports [maybe provided (runtime-defined)] [no usage info] */
/*! runtime requirements: */
eval("console.log(\"hello webpack\");\n\n//# sourceURL=webpack:///./index.js?");
}();
/******/ })()
代码太多了,我就不全部展示了,可以看到,把babel-polyfill的代码直接包含进来了。
对象
我们可以使用一个对象去描述入口文件
module.exports = {
//...
entry: {
home: './home.js',
shared: ['react', 'react-dom', 'redux', 'react-redux'], //共享的entry配置
catalog: {
import: './catalog.js',
filename: 'pages/catalog.js',
dependOn:'shared' //可以依赖某个entry
},
personal: {
import: './personal.js',
filename: 'pages/personal.js',
dependOn:'shared'
}
}
};
比如我们需要引入babel-polyfill,也可以改成以下配置:
webpack.config.js:
const path = require("path");
module.exports = {
mode: "development",
context: path.resolve(__dirname, "./src"),
// entry: ["babel-polyfill","./index.js"]
entry: {
shared: ["babel-polyfill"],
mainStr: "./index.js",
mainArray: ["./index.js"],
main: {
import: "./index.js",
dependOn: "shared",
filename: 'pages/main_polyfill.js',
}
}
};
很尴尬😂,我当前源码的版本没有这样的功能,估计webpack5.0发布完后会有的,
Xxx/webpack-src/lib/EntryOptionPlugin.js:
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
"use strict";
const DynamicEntryPlugin = require("./DynamicEntryPlugin");
const EntryPlugin = require("./EntryPlugin");
/** @typedef {import("../declarations/WebpackOptions").EntryItem} EntryItem */
/** @typedef {import("./Compiler")} Compiler */
module.exports = class EntryOptionPlugin {
/**
* @param {Compiler} compiler the compiler instance one is tapping into
* @returns {void}
*/
apply(compiler) {
compiler.hooks.entryOption.tap("EntryOptionPlugin", (context, entry) => {
/**
* @param {EntryItem} entry entry array or single path
* @param {string} name entry key name
* @returns {void}
*/
const applyEntryPlugins = (entry, name) => {
if (typeof entry === "string") {
new EntryPlugin(context, entry, name).apply(compiler);
} else if (Array.isArray(entry)) {
for (const item of entry) {
applyEntryPlugins(item, name);
}
}
};
if (typeof entry === "string" || Array.isArray(entry)) {
applyEntryPlugins(entry, "main");
} else if (typeof entry === "object") {
for (const name of Object.keys(entry)) {
applyEntryPlugins(entry[name], name);
}
} else if (typeof entry === "function") {
new DynamicEntryPlugin(context, entry).apply(compiler);
}
return true;
});
}
};
可以看到当前版本的webpack5.0.0-beta.7(我们这里用webpack5.0.0-beta.7分析是因为前面一篇文章的源码分析用的是这个版本)是不支持这种配置的,但是在最新的webpack5.0.0.-beta.21中是有的:
x x x/webpack-src/lib/EntryOptionPlugin.js(webpack5.0.0-beta.21)
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
"use strict";
const DynamicEntryPlugin = require("./DynamicEntryPlugin");
const EntryPlugin = require("./EntryPlugin");
/** @typedef {import("../declarations/WebpackOptions").EntryItem} EntryItem */
/** @typedef {import("./Compiler")} Compiler */
module.exports = class EntryOptionPlugin {
/**
* @param {Compiler} compiler the compiler instance one is tapping into
* @returns {void}
*/
apply(compiler) {
compiler.hooks.entryOption.tap("EntryOptionPlugin", (context, entry) => {
/**
* @param {EntryItem} entry entry array or single path
* @param {string} name entry key name
* @returns {void}
*/
const applyEntryPlugins = (entry, name) => {
if (typeof entry === "string") {
new EntryPlugin(context, entry, name).apply(compiler);
} else if (Array.isArray(entry)) {
for (const item of entry) {
applyEntryPlugins(item, name);
}
}
};
if (typeof entry === "string" || Array.isArray(entry)) {
applyEntryPlugins(entry, "main");
} else if (typeof entry === "object") {
for (const name of Object.keys(entry)) {
applyEntryPlugins(entry[name], name);
}
} else if (typeof entry === "function") {
new DynamicEntryPlugin(context, entry).apply(compiler);
}
return true;
});
}
};
ok,那么当前版本的webpack中怎么配置对象类型的入口呢?
const path = require("path");
module.exports = {
mode: "development",
context: path.resolve(__dirname, "./src"),
// entry: ["babel-polyfill","./index.js"]
entry: {
app: "./index.js"
}
};
动态入口
module.exports = {
//...
entry: () => './demo'
};
或者
module.exports = {
//...
entry: () => new Promise((resolve) => resolve(['./demo', './demo2']))
};
For exa
可以看到,甚至可以配置一个方法返回一个Promise对象方式,那么我们可以做什么呢? 比如我们webpack挂载在远程服务器,需要根据特殊的指定打包特定的入口,我们就可以配置一个异步的入口。
输出(output)
path
string:
默认为path.join(process.cwd(), "dist")
output 目录对应一个绝对路径。
path: path.resolve(__dirname, 'dist')
注意,[hash]
在参数中被替换为编译过程(compilation)的 hash。详细信息请查看指南 - 缓存。
对应源码位置:
lib/WebpackOptionsDefaulter.js
...
this.set("output.path", path.join(process.cwd(), "dist"));
...
我们改一下打包后输出的位置为“lib”目录:
webpack.config.js:
const path = require("path");
module.exports = {
mode: "development",
context: path.resolve(__dirname, "./src"),
// entry: ["babel-polyfill","./index.js"]
entry: {
app: "./index.js"
},
output: {
path: path.join(process.cwd(), "lib"), //默认为path.join(process.cwd(), "dist")
}
};
运行打包后文件被放在了lib目录:
lib/app.js
/******/ (() => { // webpackBootstrap
/*!******************!*\
!*** ./index.js ***!
\******************/
/*! exports [maybe provided (runtime-defined)] [no usage info] */
/*! runtime requirements: */
eval("console.log(\"hello webpack\");\n\n//# sourceURL=webpack:///./index.js?");
/******/ })()
;
pathinfo
boolean
告诉 webpack 在 bundle 中引入「所包含模块信息」的相关注释。此选项默认值是 false
,并且不应该用于生产环境(production),但是对阅读开发环境(development)中的生成代码(generated code)极其有用。
pathinfo: true
注意,这些注释也会被添加至经过 tree shaking 后生成的 bundle 中。
源码位置:
lib/WebpackOptionsDefaulter.js
可以看到,当mode为“development”的时候是开启的状态,打包完毕后会把模块的依赖信息打印出来,比如上面打包过后的lib/app.js文件中的注释:
/******/ (() => { // webpackBootstrap
/*!******************!*\
!*** ./index.js ***!
\******************/
/*! exports [maybe provided (runtime-defined)] [no usage info] */
/*! runtime requirements: */
eval("console.log(\"hello webpack\");\n\n//# sourceURL=webpack:///./index.js?");
/******/ })()
;
我们关掉之后再来测试一下:
webpack.config.js
const path = require("path");
module.exports = {
mode: "production",
context: path.resolve(__dirname, "./src"),
// entry: ["babel-polyfill","./index.js"]
entry: {
app: "./index.js"
},
output: {
path: path.join(process.cwd(), "lib"), //默认为path.join(process.cwd(), "dist")
pathinfo: false
}
};
lib/app.js:
/******/ (() => { // webpackBootstrap
eval("console.log(\"hello webpack\");\n\n//# sourceURL=webpack:///./index.js?");
/******/ })()
;
可以看到,少了模块说明的注释信息。
开发环境记得关闭哦!(webpack默认在生产模式的时候是关闭的,如果开启的话webpack会利用terser-webpack-plugin插件创建lisence信息)
比如:
webpack.config.js
const path = require("path");
module.exports = {
mode: "production",
context: path.resolve(__dirname, "./src"),
// entry: ["babel-polyfill","./index.js"]
entry: {
app: "./index.js"
},
output: {
path: path.join(process.cwd(), "lib"), //默认为path.join(process.cwd(), "dist")
pathinfo: true
}
};
打包过后会在lib下面生成两个文件,
app.js.LICENSE:
/*!******************!*\
!*** ./index.js ***!
\******************/
/*! exports [maybe provided (runtime-defined)] [unused] */
/*! runtime requirements: */
/*! ModuleConcatenation bailout: Module is not an ECMAScript module */
app.js:
/*! For license information please see app.js.LICENSE */
console.log("hello webpack");
pathinfo更多用法大家可以自己去看源码哦。
publichPath
string = ''
function
对于按需加载(on-demand-load)或加载外部资源(external resources)(如图片、文件等)来说,output.publicPath 是很重要的选项。如果指定了一个错误的值,则在加载这些资源时会收到 404 错误。
此选项指定在浏览器中所引用的「此输出目录对应的公开 URL」。相对 URL(relative URL) 会被相对于 HTML 页面(或 `` 标签)解析。相对于服务的 URL(Server-relative URL),相对于协议的 URL(protocol-relative URL) 或绝对 URL(absolute URL) 也可是可能用到的,或者有时必须用到,例如:当将资源托管到 CDN 时。
该选项的值是以 runtime(运行时) 或 loader(载入时) 所创建的每个 URL 为前缀。因此,在多数情况下,此选项的值都会以/
结束。
默认值是一个空字符串 ""
。
我们写一个异步chunk,首先我们在src目录底下创建一个demo-publicpath.js文件:
src/demo-publicpath.js
export function say() {
document.body.append(document.createTextNode("hello webpack"))
}
代码很简单,就是导出一个方法,然后在body中拼了一个文本节点。
在index.js中引用demo-publicpath.js模块的say方法,
src/index.js:
import("./demo-publicpath").then((demoPublicPath) => {
demoPublicPath.say();
});
ok,可以看到,我们用了es的一个动态导入,然后webpack判断是import动态导入会采用jsonp的形式加载对应的模块。
看一下配置文件,
webpack.config.js:
const path = require("path");
module.exports = {
mode: "development",
context: path.resolve(__dirname, "./src"),
// entry: ["babel-polyfill","./index.js"]
entry: {
app: "./index.js"
},
output: {
path: path.join(process.cwd(), "lib"), //默认为path.join(process.cwd(), "dist")
pathinfo: true,
}
};
打包编译:
192:webpack-demo yinqingyang$ npx webpack
Hash: 1f5ccbb68967d389dee1
Version: webpack 5.0.0-beta.9
Time: 132ms
Built at: 2020-07-07 20:48:31
Asset Size
app.js 8.97 KiB [compared for emit] [name: app]
demo-publicpath_js.js 952 bytes [emitted]
Entrypoint app = app.js
./index.js 84 bytes [built]
./demo-publicpath.js 92 bytes [built]
+ 7 hidden modules
192:webpack-demo yinqingyang$
可以看到,打包过后生成了两个文件,
lib/app.js:
/******/ (() => { // webpackBootstrap
/******/ var __webpack_modules__ = ({});
/************************************************************************/
/******/ // The module cache
/******/ var __webpack_module_cache__ = {};
/******/
/******/ // The require function
/******/ function __webpack_require__(moduleId) {
/******/ // Check if module is in cache
/******/ if(__webpack_module_cache__[moduleId]) {
/******/ return __webpack_module_cache__[moduleId].exports;
/******/ }
/******/ // Create a new module (and put it into the cache)
/******/ var module = __webpack_module_cache__[moduleId] = {
/******/ // no module.id needed
/******/ // no module.loaded needed
/******/ exports: {}
/******/ };
/******/
/******/ // Execute the module function
/******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__);
/******/
/******/ // Return the exports of the module
/******/ return module.exports;
/******/ }
/******/
/******/ // expose the modules object (__webpack_modules__)
/******/ __webpack_require__.m = __webpack_modules__;
/******/
/************************************************************************/
/******/ /* webpack/runtime/define property getters */
/******/ (() => {
/******/ // define getter functions for harmony exports
/******/ __webpack_require__.d = (exports, definition) => {
/******/ for(var key in definition) {
/******/ if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {
/******/ Object.defineProperty(exports, key, { enumerable: true, get: definition[key] });
/******/ }
/******/ }
/******/ };
/******/ })();
/******/
/******/ /* webpack/runtime/ensure chunk */
/******/ (() => {
/******/ __webpack_require__.f = {};
/******/ // This file contains only the entry chunk.
/******/ // The chunk loading function for additional chunks
/******/ __webpack_require__.e = (chunkId) => {
/******/ return Promise.all(Object.keys(__webpack_require__.f).reduce((promises, key) => {
/******/ __webpack_require__.f[key](chunkId, promises);
/******/ return promises;
/******/ }, []));
/******/ };
/******/ })();
/******/
/******/ /* webpack/runtime/get javascript chunk filename */
/******/ (() => {
/******/ // This function allow to reference async chunks
/******/ __webpack_require__.u = (chunkId) => {
/******/ // return url for filenames based on template
/******/ return "" + chunkId + ".js";
/******/ };
/******/ })();
/******/
/******/ /* webpack/runtime/hasOwnProperty shorthand */
/******/ (() => {
/******/ __webpack_require__.o = (obj, prop) => Object.prototype.hasOwnProperty.call(obj, prop)
/******/ })();
/******/
/******/ /* webpack/runtime/make namespace object */
/******/ (() => {
/******/ // define __esModule on exports
/******/ __webpack_require__.r = (exports) => {
/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
/******/ }
/******/ Object.defineProperty(exports, '__esModule', { value: true });
/******/ };
/******/ })();
/******/
/******/ /* webpack/runtime/publicPath */
/******/ (() => {
/******/ __webpack_require__.p = "./lib/";
/******/ })();
/******/
/******/ /* webpack/runtime/jsonp chunk loading */
/******/ (() => {
/******/
/******/
/******/ // object to store loaded and loading chunks
/******/ // undefined = chunk not loaded, null = chunk preloaded/prefetched
/******/ // Promise = chunk loading, 0 = chunk loaded
/******/ var installedChunks = {
/******/ "app": 0
/******/ };
/******/
/******/
/******/
/******/
/******/ __webpack_require__.f.j = (chunkId, promises) => {
/******/ // JSONP chunk loading for javascript
/******/ var installedChunkData = __webpack_require__.o(installedChunks, chunkId) ? installedChunks[chunkId] : undefined;
/******/ if(installedChunkData !== 0) { // 0 means "already installed".
/******/
/******/ // a Promise means "currently loading".
/******/ if(installedChunkData) {
/******/ promises.push(installedChunkData[2]);
/******/ } else {
/******/ if(true) { // all chunks have JS
/******/ // setup Promise in chunk cache
/******/ var promise = new Promise((resolve, reject) => {
/******/ installedChunkData = installedChunks[chunkId] = [resolve, reject];
/******/ });
/******/ promises.push(installedChunkData[2] = promise);
/******/
/******/ // start chunk loading
/******/ var url = __webpack_require__.p + __webpack_require__.u(chunkId);
/******/ var loadingEnded = () => {
/******/ if(__webpack_require__.o(installedChunks, chunkId)) {
/******/ installedChunkData = installedChunks[chunkId];
/******/ if(installedChunkData !== 0) installedChunks[chunkId] = undefined;
/******/ if(installedChunkData) return installedChunkData[1];
/******/ }
/******/ };
/******/ var script = document.createElement('script');
/******/ var onScriptComplete;
/******/
/******/ script.charset = 'utf-8';
/******/ script.timeout = 120;
/******/ if (__webpack_require__.nc) {
/******/ script.setAttribute("nonce", __webpack_require__.nc);
/******/ }
/******/ script.src = url;
/******/
/******/ // create error before stack unwound to get useful stacktrace later
/******/ var error = new Error();
/******/ onScriptComplete = (event) => {
/******/ onScriptComplete = () => {
/******/
/******/ }
/******/ // avoid mem leaks in IE.
/******/ script.onerror = script.onload = null;
/******/ clearTimeout(timeout);
/******/ var reportError = loadingEnded();
/******/ if(reportError) {
/******/ var errorType = event && (event.type === 'load' ? 'missing' : event.type);
/******/ var realSrc = event && event.target && event.target.src;
/******/ error.message = 'Loading chunk ' + chunkId + ' failed.\n(' + errorType + ': ' + realSrc + ')';
/******/ error.name = 'ChunkLoadError';
/******/ error.type = errorType;
/******/ error.request = realSrc;
/******/ reportError(error);
/******/ }
/******/ }
/******/ ;
/******/ var timeout = setTimeout(() => {
/******/ onScriptComplete({ type: 'timeout', target: script })
/******/ }, 120000);
/******/ script.onerror = script.onload = onScriptComplete;
/******/ document.head.appendChild(script);
/******/ } else installedChunks[chunkId] = 0;
/******/
/******/ // no HMR
/******/ }
/******/ }
/******/
/******/ // no chunk preloading needed
/******/ };
/******/
/******/ // no prefetching
/******/
/******/ // no HMR
/******/
/******/ // no HMR manifest
/******/
/******/ // no deferred startup or startup prefetching
/******/
/******/ // install a JSONP callback for chunk loading
/******/ function webpackJsonpCallback(data) {
/******/ var chunkIds = data[0];
/******/ var moreModules = data[1];
/******/
/******/ var runtime = data[3];
/******/
/******/ // add "moreModules" to the modules object,
/******/ // then flag all "chunkIds" as loaded and fire callback
/******/ var moduleId, chunkId, i = 0, resolves = [];
/******/ for(;i < chunkIds.length; i++) {
/******/ chunkId = chunkIds[i];
/******/ if(__webpack_require__.o(installedChunks, chunkId) && installedChunks[chunkId]) {
/******/ resolves.push(installedChunks[chunkId][0]);
/******/ }
/******/ installedChunks[chunkId] = 0;
/******/ }
/******/ for(moduleId in moreModules) {
/******/ if(__webpack_require__.o(moreModules, moduleId)) {
/******/ __webpack_require__.m[moduleId] = moreModules[moduleId];
/******/ }
/******/ }
/******/ if(runtime) runtime(__webpack_require__);
/******/ if(parentJsonpFunction) parentJsonpFunction(data);
/******/
/******/ while(resolves.length) {
/******/ resolves.shift()();
/******/ }
/******/
/******/ };
/******/
/******/ var jsonpArray = window["webpackJsonp"] = window["webpackJsonp"] || [];
/******/ var oldJsonpFunction = jsonpArray.push.bind(jsonpArray);
/******/ jsonpArray.push = webpackJsonpCallback;
/******/
/******/ var parentJsonpFunction = oldJsonpFunction;
/******/ })();
/******/
/************************************************************************/
/*!******************!*\
!*** ./index.js ***!
\******************/
/*! unknown exports (runtime-defined) */
/*! exports [maybe provided (runtime-defined)] [no usage info] */
/*! runtime requirements: __webpack_require__.e, __webpack_require__, __webpack_require__.* */
eval("__webpack_require__.e(/*! import() */ \"demo-publicpath_js\").then(__webpack_require__.bind(__webpack_require__, /*! ./demo-publicpath */ \"./demo-publicpath.js\")).then((demoPublicPath) => {\n demoPublicPath.say();\n});\n\n//# sourceURL=webpack:///./index.js?");
/******/ })()
;
lib/demo-publicpath_js.js:
(window["webpackJsonp"] = window["webpackJsonp"] || []).push([["demo-publicpath_js"],{
/***/ "./demo-publicpath.js":
/*!****************************!*\
!*** ./demo-publicpath.js ***!
\****************************/
/*! namespace exports */
/*! export say [provided] [no usage info] [missing usage info prevents renaming] */
/*! other exports [not provided] [no usage info] */
/*! runtime requirements: __webpack_exports__, __webpack_require__.d, __webpack_require__.r, __webpack_require__.* */
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
"use strict";
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"say\": () => /* binding */ say\n/* harmony export */ });\nfunction say() {\n document.body.append(document.createTextNode(\"hello webpack\"))\n}\n\n//# sourceURL=webpack:///./demo-publicpath.js?");
/***/ })
}]);
然后我们在根目录创建一个test.html文件用来测试,
test.html:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<script src="./lib/app.js"></script>
</body>
</html>
我们到浏览器运行一下test.html,打开浏览器会发现报错信息:
app.js:160 GET http://localhost:8080/webpack-demo/demo-publicpath_js.js 404 (Not Found)
__webpack_require__.f.j @ app.js:160
(anonymous) @ app.js:50
__webpack_require__.e @ app.js:49
eval @ index.js:1
(anonymous) @ app.js:223
(anonymous) @ app.js:224
app.js:136 Uncaught (in promise) ChunkLoadError: Loading chunk demo-publicpath_js failed.
(error: http://localhost:8080/webpack-demo/demo-publicpath_js.js)
at Object.__webpack_require__.f.j (http://localhost:8080/webpack-demo/lib/app.js:136:29)
at http://localhost:8080/webpack-demo/lib/app.js:50:40
at Array.reduce (<anonymous>)
at Function.__webpack_require__.e (http://localhost:8080/webpack-demo/lib/app.js:49:67)
at eval (webpack:///./index.js?:1:21)
at http://localhost:8080/webpack-demo/lib/app.js:223:1
at http://localhost:8080/webpack-demo/lib/app.js:224:12
__webpack_require__.f.j @ app.js:136
(anonymous) @ app.js:50
__webpack_require__.e @ app.js:49
eval @ index.js:1
(anonymous) @ app.js:223
(anonymous) @ app.js:224
Promise.then (async)
eval @ index.js:1
(anonymous) @ app.js:223
(anonymous) @ app.js:22
可以看到,默认去加载http://localhost:8080/webpack-demo/demo-publicpath_js.js文件,但是我们需要加载的是http://localhost:8080/webpack-demo/lib/demo-publicpath_js.js,少了个“lib”目录,所以我们修改一下publicPath让它默认加上lib目录,
webpack.config.js:
const path = require("path");
module.exports = {
mode: "development",
context: path.resolve(__dirname, "./src"),
// entry: ["babel-polyfill","./index.js"]
entry: {
app: "./index.js"
},
output: {
path: path.join(process.cwd(), "lib"), //默认为path.join(process.cwd(), "dist")
pathinfo: true,
publicPath: "./lib/"
}
};
然后我们继续打包运行:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-rROE8USj-1594287268615)(/Users/yinqingyang/前端架构系列之(webpack)]/webpack-demo/pub1.png)
可以看到,代码正常运行并且页面显示了预期结果。
webpack在处理异步chunk的时候会动态的创建一个script标签,在上面打包完毕的app.js我们可以看到这么一段代码,
lib/app.js:
var script = document.createElement('script');
/******/ var onScriptComplete;
/******/
/******/ script.charset = 'utf-8';
/******/ script.timeout = 120;
/******/ if (__webpack_require__.nc) {
/******/ script.setAttribute("nonce", __webpack_require__.nc);
/******/ }
/******/ script.src = url;
/******/
/******/ // create error before stack unwound to get useful stacktrace later
/******/ var error = new Error();
/******/ onScriptComplete = (event) => {
/******/ onScriptComplete = () => {
/******/
/******/ }
/******/ // avoid mem leaks in IE.
/******/ script.onerror = script.onload = null;
/******/ clearTimeout(timeout);
/******/ var reportError = loadingEnded();
/******/ if(reportError) {
/******/ var errorType = event && (event.type === 'load' ? 'missing' : event.type);
/******/ var realSrc = event && event.target && event.target.src;
/******/ error.message = 'Loading chunk ' + chunkId + ' failed.\n(' + errorType + ': ' + realSrc + ')';
/******/ error.name = 'ChunkLoadError';
/******/ error.type = errorType;
/******/ error.request = realSrc;
/******/ reportError(error);
/******/ }
/******/ }
/******/ ;
/******/ var timeout = setTimeout(() => {
/******/ onScriptComplete({ type: 'timeout', target: script })
/******/ }, 120000);
/******/ script.onerror = script.onload = onScriptComplete;
/******/ document.head.appendChild(script);
也就是动态创建了一个script标签用来异步加载demo-publicpath_js.js文件,但是scirpt标签的src是啥呢?
url=publicPath+chunkPath;
因为publicPath
默认为空字符串,所以转成script标签就为:
<script src="demo-publicpath_js.js"></script>
这样是加载不到我们的demo-publicpath_js.js文件的,因为我们默认打包到lib目录了。
所以当我们修改了publicPath
为./lib/
后,转成script标签就为:
<script src="./lib/demo-publicpath_js.js"></script>
这样就能正常加载到了。
当然,publicPath
也可以为一个完整的cdn域名地址,比如:“http://localhost:8080/webpack-demo/lib/”,转成script标签就为:
<script src="http://localhost:8080/webpack-demo/lib/demo-publicpath_js.js"></script>
在编译时(compile time)无法知道输出文件的 publicPath
的情况下,可以留空,然后在入口文件(entry file)处使用自由变量(free variable) __webpack_public_path__
,以便在运行时(runtime)进行动态设置。
比如,我们去掉配置文件的publicPath配置,
webpack.config.js:
const path = require("path");
module.exports = {
mode: "development",
context: path.resolve(__dirname, "./src"),
// entry: ["babel-polyfill","./index.js"]
entry: {
app: "./index.js"
},
output: {
path: path.join(process.cwd(), "lib"), //默认为path.join(process.cwd(), "dist")
pathinfo: true,
}
};
然后在入口文件index.js中添加以下代码,
src/index.js:
__webpack_public_path__ = "http://localhost:8080/webpack-demo/lib/";
import("./demo-publicpath").then((demoPublicPath) => {
demoPublicPath.say();
});
我们编译打包然后运行test.html文件,会发现也是一样的结果,效果我就不截图演示了。
filename
string
function (pathData, assetInfo) => string
此选项决定了每个输出 bundle 的名称。这些 bundle 将写入到 output.path
选项指定的目录下。
默认为[name].js
,[name]
代表的就是chunk的名称,除了name
外filename可选参数还可以为:
以下就是所有的可选参数 (主要由webpack的TemplatedPathPlugin`插件实现):
Template | Description |
---|---|
[hash] | 模块唯一标识符的hash值 |
[contenthash?:len] | the hash of the content of a file, which is different for each asset |
[chunkhash?:len] | The hash of the chunk content |
[name] | The module name |
[id] | The module identifier |
[query] | 跟在文件名“?”符号后面的字符串 |
[function] | The function, which can return filename [ string ] |
源码地址:lib/TemplatedPathPlugin.js
我们修改一些配置文件,
const path = require("path");
module.exports = {
mode: "development",
context: path.resolve(__dirname, "./src"),
// entry: ["babel-polyfill","./index.js"]
entry: {
app: "./index.js"
},
output: {
path: path.join(process.cwd(), "lib"), //默认为path.join(process.cwd(), "dist")
pathinfo: true,
filename: "[name].[contenthash:16].[fullhash:16].[id].js"
}
};
然后我们编译运行后会发现lib目录生成了两个文件:
app.d61bd5c5dcae317f.83aff0fe69484673.app.js
demo-publicpath_js.d29ed06019743d38.83aff0fe69484673.demo-publicpath_js.js
那为什么id
跟name
是一样的值呢? 因为我们使用的mode
是“development”,webpack默认对id使用的是NamedModuleIdsPlugin插件(直接使用module的名称代替id),如果mode
使用的是"production"的话webpack就会用HashedModuleIdsPlugin插件处理id,我们直接修改配置文件把mode
改成“production”,
webpack.config.js:
const path = require("path");
module.exports = {
mode: "production",
context: path.resolve(__dirname, "./src"),
// entry: ["babel-polyfill","./index.js"]
entry: {
app: "./index.js"
},
output: {
path: path.join(process.cwd(), "lib"), //默认为path.join(process.cwd(), "dist")
pathinfo: true,
filename: "[name].[contenthash:16].[fullhash:16].[id].js",
}
};
然后我们编译运行后会发现lib目录生成了四个文件:
721.47800a7d336db006.4eae5adf4c890bf1.721.js
721.47800a7d336db006.4eae5adf4c890bf1.721.js.LICENSE
app.c0c0395ab8956721.4eae5adf4c890bf1.143.js
app.c0c0395ab8956721.4eae5adf4c890bf1.143.js.LICENSE
可以看到,入口文件打包后是预期效果了,但是chunk的name跟id还是相同的,还是webpack默认配置的问题,当使用production的时候,为了防止chunk名称重复就直接把name换成了id。