5分钟打造简易高效的webpack常用配置
发布时间 - 2026-01-11 02:10:50 点击率:次webpack给前端开发带来了毋庸置疑的改变,它把JS,图片,css都作为模块处理,同时具有开发便捷,自动化,兼容AMD写法等等诸多无须赘述的优点,更令人称道的是其插件社区非常强大,对于不同的业务需求和技术需求社区都有大量插件可供使用。

凡事都具有两面性,许多人说:前端开发再也不能只需新建HTML文件和JS文件就可以开始写代码了。webpack带来了更高级更规范的前端开发模式,由于其本身也在不断完善中,从1到2再到发布不久的webpack3,频繁的修改给新手带来了许多困惑。而且网络上各种教程鱼目混杂,经常出现别人的教程代码copy下来在自己的环境却跑不通的蛋疼问题。就拿 devtool 配置项来说,官方文档提供了多达7种的配置方法,连react核心团队成员Pete Hunt都在twitter上调侃:我分不清webpack的许多配置之间的区别。所以今天我们抛开那些琳琅满目的插件和令人烦躁的配置项,笔者和大家一起5分钟从零搭建一个简易高效的webpack开发环境。
首先我们明确一下需求:
- 打包调试
- 提取公共代码
- 压缩
- 热替换
1.打包调试
第一步,我们在目标文件夹下安装webpack(假设已有 package.json )
npm i webpack@ -g cnpm i webpack@ --save-dev
(这里推荐大家安装稳定的2.x版本)
项目结构如图:
我们将编写的js代码和样式文件放置在 app 文件夹内(正常项目开发需要 js 文件和 less文件更规范的组织文件结构,此处仅为演示方便)。
第二步,我们在目标文件夹下新建 webpack.config.js
module.exports = {
entry:{
main:__dirname + '/app/main.js',
},
output:{
path:__dirname + '/public',
filename:'[name].[id].js',//此格式写法后续会提到为什么
publicPath:'/public/'
}
}
我们已经完成了webpack最基础的部分:添加了文件的输入和输出。入口是 app 文件夹内的 main.js 文件,出口为 public 文件夹。接下来我们来处理各种文件的解析,就是大名鼎鼎的 loader 的舞台了。假设我们使用 es6 和 less 开发,那么我们需要:
npm i babel-loader babel-core babel-preset-es2015 babel-preset-stage-0 --save-dev npm i less less-loader css-loader style-loader --save-dev
接下来我们只需要在 modules 字段下把这些 loader 加进去:
module.exports = {
devtool:'cheap-module-eval-source-map',//多种选择,选择最适合自己的
entry:{
main:__dirname + '/app/main.js',
},
output:{
path:__dirname + '/public',
filename:'[name].[id].js',
publicPath:'/public/'
},
module:{
loaders:[
{
test:/\.js$/, //解析文件类型
exclude:/node_modules/, //排除node_modules文件
loader:'babel-loader', //使用哪种loader解析
query:{
presets:['es2015','stage-0']//loader的配置项,解析es6
}
},
{
test:/\.less$/,
exclude:/node_modules/,
loader:'style-loader!css-loader!less-loader'//顺序为从右向左
}
]
},
}
大功告成!
如果你在全局安装有webpack的话,可以在终端敲入webpack并回车,几秒钟后, main.js 文件已经在 public 打包出来了!
之后我们在 index.html 中引入 main.0.js 文件,再打开 index.html 就可以看到效果了。
以上步骤,我们已经实现了文件的打包调试,但是现在有个问题摆在我们面前:第三方库代码和业务代码打包到了同一个文件 main.0.js 内,每次更新代码都要更新整个文件。那么接下来我们对代码进行拆分。
2.提取公共代码
引入 CommonsChunkPlugin 插件,在 webpack.config.js 添加如下内容:
module.exports = {
devtool:'cheap-module-eval-source-map',
entry:{
main:__dirname + '/app/main.js',
vendor:'moment'
},
output:{
path:__dirname + '/public',
filename:'[name].[id].js',
publicPath:'/public/'
},
module:{
loaders:[
{
test:/\.js$/,
exclude:/node_modules/,
loader:'babel-loader',
query:{
presets:['es2015','stage-0']
}
},
{
test:/\.less$/,
exclude:/node_modules/,
loader:'style-loader!css-loader!less-loader'
}
]
},
plugins:[
new webpack.optimize.CommonsChunkPlugin({
names:['vendor','manifest']
})
]
}
我们看到向插件的构造函数传入了两个参数 vendor 和 manifest ,以及我们在 entry 也加入了新的入口 moment 。 moment 是常用的时间处理的第三方库,我们可以通过 npm i moment --save-dev 进行安装。而 entry 处的 vendor 将成为 output 字段 filename 中 [name] 的值,也就是说将打包出 main.x.js 和 vendor.x.js 两个文件, main.x.js 文件将保存我们的业务代码, vendor.x.js 将保存 moment 的代码,这样我们将公共代码和业务代码进行了初步分离。
在新添加的 CommonmChunkPlugin 插件中,我们添加了 manifest 值,这是为什么呢?如果你不添加这个值,你在打包时会发现, main.x.js 有更新, vendor.x.js 还是有更新,并未真正实现"分离"。官方文档对此的解释是:
The issue here is that on every build, webpack generates some webpack runtime code, which helps webpack do it's job. When there is a single bundle, the runtime code resides in it. But when multiple bundles are generated, the runtime code is extracted into the common module, here the vendor file.
大致的意思就是说,webpack每次编译时运行的代码会影响到 hash 值的变化,当只有一个打包文件时这部分代码会塞进去,当有多个打包文件时,这部分代码会进入公共的 vendor 。所以解决办法是使用 manifest 字段把这部分代码从 vendor 中作为一个公共模块抽出来,从而不会影响 vendor 。
将以上的配置写入 webpack.config.js ,运行webpack命令,我们发现业务代码和公共库代码成功分离,改写 main.1.js 文件的内容,再次打包,发现 vendor 文件并没有变化,成功!
当我们再进行打包时,发现又会多出了新的 main.x.js 等文件,打包三次就会出现三个 main.x.js 文件,此时该怎么办呢?我们可以使用 clean-webpack-plugin 插件:
npm i clean-webpack-plugin --save-dev
然后在 webpack.config.js 中引入:
var CleanWebpackPlugin = require('clean-webpack-plugin');
new CleanWebpackPlugin(
['public/main.*.js','public/manifest.*.js'],//要删除的文件目录匹配
{
root:__dirname,
verbose:true,
dry:false
}
),
这样我们每次在打包新的代码时,旧文件就会删除,不会再出现同一份文件存在多份的情况。
3.压缩
在webpack中,图片,css,js等等其他资源皆可压缩,本文仅以压缩js为例。
安装插件:
npm i uglifyjs-webpack-plugin --save-dev
在 webpack.config.js 中引入:
var UglifyJsPlugin = require('uglifyjs-webpack-plugin');
new UglifyJsPlugin({
beautify:true,
exclude:['/node_modules/'],
compress:{
warnings:false
},
output:{
comments:false
}
})
我们指定了压缩的方法,排除了不需要压缩的 node_modules 部分,同时我们去除了 comments 部分( comments 为@license等注释,是可观的压缩空间)。再次在终端输入打包命令,可见js打包后的体积有令人满意的减小。
4.热替换
webpack总是绕不开热替换的话题。热替换的功能配置和原理是一大话题,三天三夜也说不完,也并非本文重点,本文只提供简易高效的配置方法。
热替换存在两种使用方式, cli 和 node 。 cli 方式无需添加新的热替换插件,且无需在入口处添加 webpack-dev-server 等入口,故本文采用 cli 使用方式。
在 webpack.config.js 中添加 devServer 字段,加入如下代码:
devServer:{
inline:true,
hot:true
},
保存后运行 webpack-dev-server --inline --hot --progress ,再修改下 main.less 文件的样式,会发现浏览器并没有刷新,但页面已经发生了变化,我们的热替换功能也成功加入了!
tips:
在实际项目打包时,可以将 filename 字段的值换为 [name].[chunkhash].js ,其中 [chunkhash] 为webpack每次打包后给每个模块的标识值,这个值每次打包后都会更换。为什么在此处我们使用 [id] 呢,因为 chunkhash 与热替换存在冲突,终端会有报错,那么使用 id 可以算作一个解决方案。这就引申出另一话题,我们可以使用两套webpack配置分别用于生产环境和开发环境,通过webpack指定config来进行打包。例如我们在开发环境使用 id ,在生产环境去掉热替换并使用 hash 的方式。而且,一些压缩插件也没必要在开发环境过度使用,两套配置能让webpack发挥最大的威力。
另外, chunkhash 和 hash 有区别, chunkhash 顾名思义是模块的标识,而 hash 是webpack每次编译的标识值,不同的资源如js和css存在 chunkhash 解耦的问题,此处不进行过多讨论。
5.运行
我们知道,每次打包后,都会有新的 main.x.js 文件生成,其hash值每次打包后都会发生变化,难道我们的 index.html 文件需要每次打包后都手动修改 main.x.js 的路径吗?还好社区提供了 html-webpack-plugin 插件,可以在已有html模板的条件下自动为我们生成带有最新代码的html文件:
npm i html-webpack-plugin --save-dev
在 webpack.config.js 中引入:
var HtmlWebpackPlugin = require('html-webpack-plugin');
new HtmlWebpackPlugin({
title:'demo',
template:'index.html'
}),
在终端运行打包命令,我们看到 public 文件夹下生成了新的 index.html 文件:
以后我们再进行调试时,以本文为例,则需要打开 localhost:8080/public/index.html ,因为每次webpack的 HtmlWebpackPlugin 都会把新的js文件加入到这个html文件内。在开发全部完成后,我们可以将js路径写死,添加到原有的 index.html 文件中。
以下是我们 webpack.config.js 全部的配置;
var webpack = require('webpack');
var CleanWebpackPlugin = require('clean-webpack-plugin');
var HtmlWebpackPlugin = require('html-webpack-plugin');
var UglifyJsPlugin = require('uglifyjs-webpack-plugin');
module.exports = {
devtool:'cheap-module-eval-source-map',
entry:{
main:__dirname + '/app/main.js',
vendor:'moment'
},
output:{
path:__dirname + '/public',
filename:'[name].[id].js',
publicPath:'/public/'
},
devServer:{
inline:true,
hot:true
},
module:{
loaders:[
{
test:/\.js$/,
exclude:/node_modules/,
loader:'babel-loader',
query:{
presets:['es2015','stage-0']
}
},
{
test:/\.less$/,
exclude:/node_modules/,
loader:'style-loader!css-loader!less-loader'
}
]
},
plugins:[
new CleanWebpackPlugin(
['public/main.*.js','public/manifest.*.js'],
{
root:__dirname,
verbose:true,
dry:false
}
),
new webpack.optimize.CommonsChunkPlugin({
names:['vendor','manifest']
}),
new HtmlWebpackPlugin({
title:'demo',
template:'index.html'
}),
new UglifyJsPlugin({
beautify:true,
exclude:['/node_modules/'],
compress:{
warnings:false
},
output:{
comments:false
}
})
]
}
整个项目,我们在 app 文件下的 main.js 内写业务代码, main.less 写样式,在 public/index.html 下使用热替换进行调试,打包后的压缩文件在 public 文件夹下,并且对业务代码,第三方代码进行了清晰地区分。
使用这份webpack配置,我们实现了:
- 工程的打包调试
- 公共代码提取,提高开发效率
- 资源压缩
- 热替换
这份配置麻雀虽小,五脏俱全。本文还有许多不完善之处,比如一些插件的使用方法,原理没有与大家讲清楚,但webpack实在太庞大了,一个插件的使用方法和原理都可以写上千字的文章了,学习不可浅尝辄止,但也不能太钻牛角尖,与大家共勉~
# webpack
# 配置
# webpack配置文件
# webpack常用配置
# webpack常用配置项配置文件介绍
# 深入理解Webpack 中路径的配置
# webpack教程之webpack.config.js配置文件
# vue-cli的webpack模板项目配置文件分析
# webpack引入eslint配置详解
# webpack多入口文件页面打包配置详解
# Vue + Webpack + Vue-loader学习教程之相关配置篇
# 详解vue2.0脚手架的webpack 配置文件分析
# 详解vue-cli + webpack 多页面实例配置优化方法
# 详解webpack之scss和postcss-loader的配置
# webpack配置sass模块的加载的方法
# webpack构建vue项目的详细教程(配置篇)
# webpack配置导致字体图标无法显示的解决方法
# webpack3+React 的配置全解
# 浅谈在vue中用webpack打包之后运行文件的问题以及相关配置方法
# 详解Webpack + ES6 最新环境搭建与配置
# webpack配置打包后图片路径出错的解决
# webpack高级配置与优化详解
# 这部
# 带来了
# 自己的
# 第三方
# 就会
# 会有
# 已有
# 你在
# 只需
# 要在
# 我们可以
# 这份
# 可以使用
# 为例
# 两套
# 进行了
# 就可以
# 的是
# 这是
# 实现了
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
如何获取PHP WAP自助建站系统源码?
国美网站制作流程,国美电器蒸汽鍋怎么用官方网站?
Laravel事件和监听器如何实现_Laravel Events & Listeners解耦应用的实战教程
Laravel怎么进行数据库事务处理_Laravel DB Facade事务操作确保数据一致性
韩国服务器如何优化跨境访问实现高效连接?
*服务器网站为何频现安全漏洞?
企业在线网站设计制作流程,想建设一个属于自己的企业网站,该如何去做?
Python图片处理进阶教程_Pillow滤镜与图像增强
Laravel如何使用Spatie Media Library_Laravel图片上传管理与缩略图生成【步骤】
HTML 中如何正确使用模板变量为元素的 name 属性赋值
uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址
黑客如何利用漏洞与弱口令入侵网站服务器?
米侠浏览器网页图片不显示怎么办 米侠图片加载修复
Python进程池调度策略_任务分发说明【指导】
如何快速生成凡客建站的专业级图册?
如何在云服务器上快速搭建个人网站?
如何用wdcp快速搭建高效网站?
图册素材网站设计制作软件,图册的导出方式有几种?
iOS中将个别页面强制横屏其他页面竖屏
制作网站软件推荐手机版,如何制作属于自己的手机网站app应用?
如何在HTML表单中获取用户输入并结合JavaScript动态控制复利计算循环
Laravel如何实现URL美化Slug功能_Laravel使用eloquent-sluggable生成别名【方法】
微信小程序 scroll-view组件实现列表页实例代码
Laravel怎么防止CSRF攻击_Laravel CSRF保护中间件原理与实践
高配服务器限时抢购:企业级配置与回收服务一站式优惠方案
详解jQuery停止动画——stop()方法的使用
HTML透明颜色代码怎么让下拉菜单透明_下拉菜单透明背景指南【技巧】
如何在 Telegram Web View(iOS)中防止键盘遮挡底部输入框
在线ppt制作网站有哪些软件,如何把网页的内容做成ppt?
Laravel如何生成URL和重定向?(路由助手函数)
Laravel安装步骤详细教程_Laravel环境搭建指南
手机怎么制作网站教程步骤,手机怎么做自己的网页链接?
Laravel如何使用Guzzle调用外部接口_Laravel发起HTTP请求与JSON数据解析【详解】
三星网站视频制作教程下载,三星w23网页如何全屏?
Laravel如何创建自定义Facades?(详细步骤)
如何在Windows 2008云服务器安全搭建网站?
儿童网站界面设计图片,中国少年儿童教育网站-怎么去注册?
如何在建站宝盒中设置产品搜索功能?
html5如何实现懒加载图片_ intersectionobserver api用法【教程】
Laravel如何使用Eloquent ORM进行数据库操作?(CRUD示例)
如何快速搭建安全的FTP站点?
中山网站制作网页,中山新生登记系统登记流程?
如何用免费手机建站系统零基础打造专业网站?
如何基于云服务器快速搭建个人网站?
七夕网站制作视频,七夕大促活动怎么报名?
Laravel如何正确地在控制器和模型之间分配逻辑_Laravel代码职责分离与架构建议
EditPlus中的正则表达式 实战(1)
Laravel如何使用Facades(门面)及其工作原理_Laravel门面模式与底层机制
Android自定义控件实现温度旋转按钮效果
Laravel如何实现邮箱地址验证功能_Laravel邮件验证流程与配置

