Plugin

何为插件(Plugin)?专注处理 webpack 在编译过程中的某个特定的任务的功能模块,可以称为插件。plugin 是一个扩展器,它丰富了 webpack 本身,针对是 loader 结束后,webpack 打包的整个过程,它并不直接操作文件,而是基于事件机制工作,会监听 webpack 打包过程中的某些节点,执行广泛的任务。

Plugin 的特点

  • 是一个独立的模块

  • 模块对外暴露一个 js 函数

  • 函数的原型 (prototype) 上定义了一个注入 compiler 对象的 apply方法 apply 函数中需要有通过 compiler 对象挂载的 webpack 事件钩子,钩子的回调中能拿到当前编译的 compilation 对象,如果是异步编译插件的话可以拿到回调 callback

  • 完成自定义子编译流程并处理 complition 对象的内部数据

  • 如果异步编译插件的话,数据处理完成后执行 callback 回调。

下面介绍 18 个常用的 webpack 插件

HotModuleReplacementPlugin

模块热更新插件。Hot-Module-Replacement 的热更新是依赖于 webpack-dev-server,后者是在打包文件改变时更新打包文件或者 reload 刷新整个页面,HRM 是只更新修改的部分。

HotModuleReplacementPluginwebpack模块自带的,所以引入webpack后,在plugins配置项中直接使用即可。

const webpack = require('webpack')plugins: [new webpack.HotModuleReplacementPlugin(), // 热更新插件
]

html-webpack-plugin

生成 html 文件。将 webpack 中entry配置的相关入口 chunkextract-text-webpack-plugin抽取的 css 样式 插入到该插件提供的template或者templateContent配置项指定的内容基础上生成一个 html 文件,具体插入方式是将样式link插入到head元素中,script插入到head或者body中。

const HtmlWebpackPlugin = require('html-webpack-plugin')plugins: [new HtmlWebpackPlugin({filename: 'index.html',template: path.join(__dirname, '/index.html'),minify: {// 压缩HTML文件removeComments: true, // 移除HTML中的注释collapseWhitespace: true, // 删除空白符与换行符minifyCSS: true, // 压缩内联css},inject: true,}),
]

inject 有四个选项值

  • true:默认值,script 标签位于 html 文件的 body 底部

  • body:script 标签位于 html 文件的 body 底部(同 true)

  • head:script 标签位于 head 标签内

  • false:不插入生成的 js 文件,只是单纯的生成一个 html 文件

多页应用打包

有时,我们的应用不一定是一个单页应用,而是一个多页应用,那么如何使用 webpack 进行打包呢。

const path = require('path')
const HtmlWebpackPlugin = require('html-webpack-plugin')
module.exports = {entry: {index: './src/index.js',login: './src/login.js',},output: {path: path.resolve(__dirname, 'dist'),filename: '[name].[hash:6].js',},//...plugins: [new HtmlWebpackPlugin({template: './public/index.html',filename: 'index.html', //打包后的文件名}),new HtmlWebpackPlugin({template: './public/login.html',filename: 'login.html', //打包后的文件名}),],
}

如果需要配置多个 HtmlWebpackPlugin,那么 filename 字段不可缺省,否则默认生成的都是 index.html

但是有个问题,index.htmllogin.html 会发现,都同时引入了 index.f7d21a.jslogin.f7d21a.js,通常这不是我们想要的,我们希望 index.html 中只引入 index.f7d21a.jslogin.html 只引入 login.f7d21a.js

HtmlWebpackPlugin 提供了一个 chunks 的参数,可以接受一个数组,配置此参数仅会将数组中指定的 js 引入到 html 文件中

module.exports = {//...plugins: [new HtmlWebpackPlugin({template: './public/index.html',filename: 'index.html', //打包后的文件名chunks: ['index'],}),new HtmlWebpackPlugin({template: './public/login.html',filename: 'login.html', //打包后的文件名chunks: ['login'],}),],
}

这样执行 npm run build,可以看到 index.html 中仅引入了 index 的 js 文件,而 login.html 中也仅引入了 login 的 js 文件。

clean-webpack-plugin

clean-webpack-plugin 用于在打包前清理上一次项目生成的 bundle 文件,它会根据 output.path 自动清理文件夹;这个插件在生产环境用的频率非常高,因为生产环境经常会通过 hash 生成很多 bundle 文件,如果不进行清理的话每次都会生成新的,导致文件夹非常庞大。

const { CleanWebpackPlugin } = require('clean-webpack-plugin')plugins: [new HtmlWebpackPlugin({template: path.join(__dirname, '/index.html'),}),new CleanWebpackPlugin(), // 所要清理的文件夹名称
]

extract-text-webpack-plugin

将 css 成生文件,而非内联 。该插件的主要是为了抽离 css 样式,防止将样式打包在 js 中引起页面样式加载错乱的现象

const ExtractTextPlugin = require('extract-text-webpack-plugin')plugins: [// 将css分离到/dist文件夹下的css文件夹中的index.cssnew ExtractTextPlugin('css/index.css'),
]

mini-css-extract-plugin

将 CSS 提取为独立的文件的插件,对每个包含 css 的 js 文件都会创建一个 CSS 文件,支持按需加载 css 和 sourceMap。只能用在 webpack4 中,对比另一个插件 extract-text-webpack-plugin 有以下特点:

  • 异步加载

  • 不重复编译,性能更好

  • 更容易使用

  • 只针对 CSS

这个插件应该只用在生产环境配置,并且在 loaders 链中不使用 style-loader, 而且这个插件暂时不支持 HMR

const MiniCssExtractPlugin = require('mini-css-extract-plugin')module.exports = {module: {rules: [{test: /\.(le|c)ss$/,use: [{loader: MiniCssExtractPlugin.loader,options: {publicPath: '../',},},'css-loader','postcss-loader','less-loader',],},],},plugins: [new MiniCssExtractPlugin({filename: 'css/[name].[contenthash:8].css',chunkFilename: 'css/[id].[contenthash:8].css',}),],
}

purifycss-webpack

有时候我们 css 写得多了或者重复了,这就造成了多余的代码,我们希望在生产环境进行去除。

const path = require('path')
const PurifyCssWebpack = require('purifycss-webpack') // 引入PurifyCssWebpack插件
const glob = require('glob') // 引入glob模块,用于扫描全部html文件中所引用的cssmodule.exports = merge(common, {plugins: [new PurifyCssWebpack({paths: glob.sync(path.join(__dirname, 'src/*.html')),}),],
})

optimize-css-assets-webpack-plugin

我们希望减小 css 打包后的体积,可以用到 optimize-css-assets-webpack-plugin

const OptimizeCSSAssetsPlugin = require("optimize-css-assets-webpack-plugin") // 压缩css代码optimization: {minimizer: [// 压缩cssnew OptimizeCSSAssetsPlugin({})]

UglifyJsPlugin

uglifyJsPluginvue-cli 默认使用的压缩代码方式,用来对 js 文件进行压缩,从而减小 js 文件的大小,加速 load 速度。它使用的是单线程压缩代码,打包时间较慢,所以可以在开发环境将其关闭,生产环境部署时再把它打开。

const UglifyJsPlugin = require('uglifyjs-webpack-plugin')plugins: [new UglifyJsPlugin({uglifyOptions: {compress: {warnings: false}},sourceMap: true,  //是否启用文件缓存parallel: true   //使用多进程并行运行来提高构建速度})

ParallelUglifyPlugin

开启多个子进程,把对多个文件压缩的工作分别给多个子进程去完成,每个子进程其实还是通过 UglifyJS 去压缩代码,但是变成了并行执行。

const ParallelUglifyPlugin = require('webpack-parallel-uglify-plugin')plugins: [new ParallelUglifyPlugin({//cacheDir 用于配置缓存存放的目录路径。cacheDir: '.cache/',sourceMap: true,uglifyJS: {output: {comments: false,},compress: {warnings: false,},},}),
]

terser-webpack-plugin

Webpack4.0 默认是使用 terser-webpack-plugin 这个压缩插件,在此之前是使用 uglifyjs-webpack-plugin,两者的区别是后者对 ES6 的压缩不是很好,同时我们可以开启 parallel 参数,使用多进程压缩,加快压缩。

const TerserPlugin = require('terser-webpack-plugin') // 压缩js代码optimization: {minimizer: [new TerserPlugin({parallel: 4, // 开启几个进程来处理压缩,默认是 os.cpus().length - 1cache: true, // 是否缓存sourceMap: false,}),]
}

NoErrorsPlugin

报错但不退出 webpack 进程。编译出现错误时,使用 NoEmitOnErrorsPlugin 来跳过输出阶段。这样可以确保输出资源不会包含错误。

plugins: [new webpack.NoEmitOnErrorsPlugin()]

compression-webpack-plugin

所有现代浏览器都支持 gzip 压缩,启用 gzip 压缩可大幅缩减传输资源大小,从而缩短资源下载时间,减少首次白屏时间,提升用户体验。

gzip 对基于文本格式文件的压缩效果最好(如:CSS、JavaScript 和 HTML),在压缩较大文件时往往可实现高达 70-90% 的压缩率,对已经压缩过的资源(如:图片)进行 gzip 压缩处理,效果很不好。

const CompressionPlugin = require('compression-webpack-plugin')plugins: [new CompressionPlugin({// gzip压缩配置test: /\.js$|\.html$|\.css/, // 匹配文件名threshold: 10240, // 对超过10kb的数据进行压缩deleteOriginalAssets: false, // 是否删除原文件}),
]

当然,这个方法还需要后端配置支持。

DefinePlugin

我们可以通过 DefinePlugin 可以定义一些全局的变量,我们可以在模块当中直接使用这些变量,无需作任何声明,DefinePluginwebpack 自带的插件。

plugins: [new webpack.DefinePlugin({DESCRIPTION: 'This Is The Test Text.',}),
]// 直接引用
console.log(DESCRIPTION)

ProvidePlugin

自动加载模块。任何时候,当 identifier 被当作未赋值的变量时, module 就会自动被加载,并且 identifier 会被这个 module 输出的内容所赋值。这是 webpack 自带的插件。

module.exports = {resolve: {alias: {jquery: './lib/jquery',},},plugins: [//提供全局的变量,在模块中使用无需用require引入new webpack.ProvidePlugin({$: 'jquery',React: 'react',}),],
}

DLLPlugin

这是在一个额外的独立的 webpack 设置中创建一个只有 dll 的 bundle(dll-only-bundle)。这个插件会生成一个名为 manifest.json 的文件,这个文件是用来让 DLLReferencePlugin 映射到相关的依赖上去的。

「使用步骤如下」

1、在 build 下创建 webpack.dll.config.js

const path = require('path')
const webpack = require('webpack')
module.exports = {entry: {vendor: ['vue-router','vuex','vue/dist/vue.common.js','vue/dist/vue.js','vue-loader/lib/component-normalizer.js','vue','axios','echarts',],},output: {path: path.resolve('./dist'),filename: '[name].dll.js',library: '[name]_library',},plugins: [new webpack.DllPlugin({path: path.resolve('./dist', '[name]-manifest.json'),name: '[name]_library',}),// 建议加上代码压缩插件,否则dll包会比较大。new webpack.optimize.UglifyJsPlugin({compress: {warnings: false,},}),],
}

2、在 webpack.prod.conf.js 的 plugin 后面加入配置

new webpack.DllReferencePlugin({manifest: require('../dist/vendor-manifest.json'),
})

3、package.json文件中添加快捷命令(build:dll)

  "scripts": {"dev": "webpack-dev-server --inline --progress --config build/webpack.dev.conf.js","start": "npm run dev","lint": "eslint --ext .js,.vue src","build": "node build/build.js","build:dll": "webpack --config build/webpack.dll.conf.js"}

生产环境打包的时候先npm run build:dll命令会在打包目录下生成 vendor-manifest.json 文件与 vendor.dll.js 文件。然后npm run build生产其他文件。

4、根目录下的入口 index.html 加入引用

<script type="text/javascript" src="./vendor.dll.js"></script>

HappyPack

HappyPack 能让 webpack 把任务分解给多个子进程去并发的执行,子进程处理完后再把结果发送给主进程。要注意的是 HappyPackfile-loaderurl-loader 支持的不友好,所以不建议对该 loader 使用。

1、HappyPack 插件安装

npm i -D happypack

2、webpack.base.conf.js 文件对 module.rules 进行配置

module: {rules: [{test: /\.js$/,use: ['happypack/loader?id=babel'],include: [resolve('src'), resolve('test')],exclude: path.resolve(__dirname, 'node_modules'),},{test: /\.vue$/,use: ['happypack/loader?id=vue'],},]
}

3、在生产环境 webpack.prod.conf.js 文件进行配置

const HappyPack = require('happypack')
// 构造出共享进程池,在进程池中包含5个子进程
const HappyPackThreadPool = HappyPack.ThreadPool({ size: 5 })
plugins: [new HappyPack({// 用唯一的标识符id,来代表当前的HappyPack是用来处理一类特定的文件id: 'babel',// 如何处理.js文件,用法和Loader配置中一样loaders: ['babel-loader?cacheDirectory'],threadPool: HappyPackThreadPool,}),new HappyPack({id: 'vue', // 用唯一的标识符id,来代表当前的HappyPack是用来处理一类特定的文件loaders: [{loader: 'vue-loader',options: vueLoaderConfig,},],threadPool: HappyPackThreadPool,}),
]

「注意,当项目较小时,多线程打包反而会使打包速度变慢。」

copy-webpack-plugin

我们在 public/index.html 中引入了静态资源,但是打包的时候 webpack 并不会帮我们拷贝到 dist 目录,因此 copy-webpack-plugin 就可以很好地帮我做拷贝的工作了。

const CopyWebpackPlugin = require('copy-webpack-plugin')
module.exports = {plugins: [new CopyWebpackPlugin({patterns: [{from: 'public/js/*.js',to: path.resolve(__dirname, 'dist', 'js'),flatten: true,},],}),],
}

IgnorePlugin

这是 webpack 内置插件,它的作用是:忽略第三方包指定目录,让这些指定目录不要被打包进去。

比如我们要使用 moment 这个第三方依赖库,该库主要是对时间进行格式化,并且支持多个国家语言。虽然我设置了语言为中文,但是在打包的时候,是会将所有语言都打包进去的。这样就导致包很大,打包速度又慢。对此,我们可以用 IgnorePlugin 使得指定目录被忽略,从而使得打包变快,文件变小。

const Webpack = require('webpack')
plugins: [//moment这个库中,如果引用了./locale/目录的内容,就忽略掉,不会打包进去new Webpack.IgnorePlugin(/\.\/locale/, /moment/),
]

我们虽然按照上面的方法忽略了包含’./locale/'该字段路径的文件目录,但是也使得我们使用的时候不能显示中文语言了,所以这个时候可以手动引入中文语言的目录。

import moment from 'moment'//手动引入所需要的语言包
import 'moment/locale/zh-cn'moment.locale('zh-cn')let r = moment().endOf('day').fromNow()
console.log(r)

前端学习笔记????

最近花了点时间把笔记整理到语雀上了,方便同学们阅读:公众号回复笔记或者简历

转载自:https://juejin.cn/post/6844904193589772301

1. JavaScript 重温系列(22篇全)

2. ECMAScript 重温系列(10篇全)

3. JavaScript设计模式 重温系列(9篇全)

4. 正则 / 框架 / 算法等 重温系列(16篇全)

5. Webpack4 入门(上)|| Webpack4 入门(下)

6. MobX 入门(上) ||  MobX 入门(下)

7. 80+篇原创系列汇总

回复“加群”与大佬们一起交流学习~

点击“阅读原文”查看 100+ 篇原创文章

【面试】815- 面试官常问的 webpack 插件相关推荐

  1. 平面设计面试官常问的问题有哪些?

      平面设计面试官常问的问题有哪些?一般情况下都会有一个开篇的自我介绍,这里建议就用平常聊天说话的语气语速来介绍就可以,介绍自己扬长避短,多讲自己的前工作项目经验,在校获奖也可以讲.与工作岗位不强相关 ...

  2. 面试官常问的 web前端 问题(二)

    面试官常问的 web前端 问题 11-20 11.什么是响应式设计? 12.为什么我们要弃用 table 标签 13.iframe 有哪些缺点 14.meta viewport 是做什么用的,怎么写? ...

  3. 面试官常问的 web前端 问题(四)

    面试官常问的 web前端 问题 31-40 31.使用 Vue 的好处 32.MVVM 定义 33.Vue 的生命周期(重点) 34.Vue 的响应式原理 35.第一次页面加载会触发哪几个钩子? 36 ...

  4. 【拥抱大厂系列】几个面试官常问的垃圾回收器,下次面试就拿这篇文章怼回去!

    点个赞,看一看,好习惯!本文 GitHub https://github.com/OUYANGSIHAI/JavaInterview 已收录,这是我花了3个月总结的一线大厂Java面试总结,本人已拿腾 ...

  5. 几个面试官常问的垃圾回收器,下次面试就拿这篇文章怼回去!

    点击上方 好好学java ,选择 星标 公众号 重磅资讯.干货,第一时间送达 今日推荐:牛人 20000 字的 Spring Cloud 总结,太硬核了~ 先瞎比比一下,上一篇文章已经过去2个多月了, ...

  6. 面试官常问的几个问题

    问题一:请介绍一下你自己 这是面试者基本上都会被问到的问题. 一般面试者回答这个问题过于平常,只说姓名.年龄.爱好.工作经验,这些在简历上都有. 其实,面试官最希望知道的是求职者能否胜任工作,包括最强 ...

  7. 面试中人力资源部常问的问题

    面试中HR常问的问题: 1.先简单介绍一下自己,以及工作经历. 2.为什么离开原来的公司? 3.如何看待加班?(外包公司最喜欢问) 4.你认为原来的公司怎么样? 5.如何看待现在的软件行业? 6.对自 ...

  8. Python程序员面试中,被常问的七道题!(详细解题思路)

    俗话说的好,打仗之前知晓敌方的军情越多,获胜的把握也会越大,知己知彼方能百战不殆嘛!换个角度来说,如今的面试不正是和古代打战一样嘛,知晓的越多,成功几率也就越大. 如今,Python是个非常受欢迎的编 ...

  9. 爆赞!Android岗大厂面试官常问的那些问题,论程序员成长的正确姿势

    开头 昨天去面了一家公司,价值观有受到冲击. 面试官技术方面没的说,他可能是个完美主义的人,无论什么事情到了他那里好像都有解决的方案,我被说的无所适从,感觉他很厉害. 但我不能认可的是,面试官觉得加班 ...

最新文章

  1. linux 程序收到sigsegv信号_信号
  2. 2017/11/3模拟赛
  3. mysql 不通过关联查询表_mysql中多表不关联查询的实现方法详解
  4. 网络安全——ipsec
  5. .net如何引用该命名空间
  6. 做一个有趣的有意思的人
  7. (转)The POM for 0.0.1-SNAPSHOT is missing, no dependency informat
  8. php背景时间渐变,CSS3怎么实现背景颜色渐变?(图文+视频)
  9. Spring Cloud Config 使用总结
  10. 匿名函数php作用,深入理解PHP中的匿名函数
  11. Zabbix---3 监控主机内存使用率
  12. 虚拟机centos6.5 --安装jdk
  13. 申请以及集成 Stripe 的 Alipay 支付方案
  14. [No000057]一个人默默背单词,小心被传染哦
  15. python软件是什么作用,python-dotenv的用途是什么?
  16. G.Power教程 | 样本量估计
  17. matlab fft2 fftshift,matlab fft fftshift
  18. Delphi FireDAC SQLite “database is locked“ 解决办法
  19. unity商店里下载的资源保存路径
  20. Word中硬回车与软回车的区别和用法

热门文章

  1. 七零年代摄影师Robert Mapplethorpe
  2. 洛谷P2903 [USACO08MAR]麻烦的干草打包机The Loathesome Hay Baler
  3. 诺基亚Lumia630传感器并未缩水
  4. python爬取陌生人的qq空间_Python爬取QQ空间好友说说并生成词云(超详细)
  5. Python3端口扫描器
  6. C++ upper_bound()和lower_bound()(二分查找中使用)的定义,使用方法和区别
  7. 传智播客微金所项目实战移动web开发
  8. Qt快速入门系列教程目录
  9. 安卓图片分类浏览器php,Android快速实现图片浏览
  10. QQzone 项目 (SSM)