webpack的plugin原理
plugin是webpack生态的重要组成,它为用户提供了一种可以直接访问到webpack编译过程的方式。它可以访问到编译过程触发的所有关键事件。
1. 基本概念
1. 如何实现一个插件
1. plugin实际是一个类(构造函数),通过在plugins配置中实例化进行调用。
// webpack.config.js
var MyExampleWebpackPlugin = require('my-example-webpack-plugin'); module.exports = {
// ... 这里是其他配置 ...
plugins: [new MyExampleWebpackPlugin({ options: xxx })]
};
2. 它在原型对象上指定了一个apply方法,入参是compiler对象
3. 指定一个事件钩子,并调用内部提供的API
4. 完成操作后,调用webpack 提供的callback方法
// 一个 JavaScript class
class MyExampleWebpackPlugin {
// 将 `apply` 定义为其原型方法,此方法以 compiler 作为参数
apply(compiler) {
// 指定要附加到的事件钩子函数
compiler.hooks.emit.tapAsync('MyExampleWebpackPlugin',
(compilation, callback) => {// 使用 webpack 提供的 plugin API 操作构建结果
compilation.addModule(/* ... */);
callback();
}
);
}
}
2. 实现插件的背景知识
由上面的步骤可知,插件功能的实现主要依赖于compiler和complation对象,而两者都是继承自Tapable对象。它暴露三种注册监听的方法Tapable对象主要是9种钩子:
const {
SyncHook,
SyncBailHook,
SyncWaterfallHook,
SyncLoopHook,
AsyncParallelHook,
AsyncParallelBailHook,
AsyncSeriesHook,
AsyncSeriesBailHook,
AsyncSeriesWaterfallHook
} = require("tapable");
其中同步四种,异步并行两种,异步串行3种。
同步钩子进行同步操作;异步钩子中进行异步操作。
compiler和compilation中的钩子都是自称自这9种钩子。钩子的工作机制类似于浏览器的事件监听。
1)生成的钩子可以注册监听事件,其中同步钩子通过tap方法监听,异步钩子通过tapAsync(+回调函数)和tapPromise(+返回promise)进行监听。
2)还可以进行拦截,通过intercept方法。
3)对于监听事件的触发,同步钩子通过call方法; 异步钩子通过callAsync方法和promise
示例1: -SyncHook
监听事件都是按照注册的顺序依次执行
const { SyncHook } = require('tapable'); const hook = new SyncHook(['name', 'age']);
hook.tap('任意字符1', (name, age) => {
console.log(1);
return undefined;
})
hook.tap('任意字符2', (name, age) => {
console.log(2);
return true;
})
hook.tap('任意字符3', (name, age) => {
console.log(3);
})
hook.call('lyra', 18); //传入的参数必须和初始化实例时传入的参数个数相同
// 执行顺序如下
1
2
3
示例2: -SyncBailHook
只要返回非undefined值,终止监听事件的调用
示例3:-SyncWaterfallHook
监听事件如果返回非undefined值,作为下个监听事件的第一个参数;如果返回undefined,返回之前的监听事件中最近的非undefined值
const { SyncWaterfallHook } = require('tapable'); const hook = new SyncWaterfallHook(['name', 'age']);
hook.tap('任意字符1', (name, age) => {
console.log('1-->',name,age);
return 19;
})
hook.tap('任意字符2', (name, age) => {
console.log('2-->',name,age);
return undefined;
})
hook.tap('任意字符3', (name, age) => {
console.log('3-->',name,age);
return 21;
})
hook.call('lyra', 18); //传入的参数必须和初始化实例时传入的参数个数相同
// 执行顺序如下
1--> lyra 18
2--> 19 18
3--> 19 18
示例4: -SyncLoopHook
只要监听事件返回的是非undefined值,则回到该钩子的第一个监听事件从头开始执行
示例5: -AsyncSeriesHook(complier.hooks.emit)
串行异步会等前一个事件结束再执行第二个监听事件
// 串行异步,则耗时需要3s
const { AsyncSeriesHook } = require('tapable'); const hook = new AsyncSeriesHook(['name', 'age']);
hook.tapPromise('任意字符1', (name, age) => {
// 因为使用的是tapPromise监听,必须返回一个promise
console.time(1)
return new Promise(function(resolve, reject){
setTimeout(function() {
resolve(1);
}, 1000)
})
})
hook.tapPromise('任意字符2', (name, age) => {
return new Promise((resolve, reject) => {
setTimeout(function() {
resolve(1);
console.timeEnd(1)
}, 2000)
});
})
// 通过promise方法触发监听
hook.promise('lyra', 18).then(() => {
// TODO
});
示例6: - AsyncParallelHook(complier.hooks.make)
并行异步,所有监听函数同时执行
// 并行异步,耗时2秒,其实就是所有监听事件中耗时最长的事件
const { AsyncParallelHook } = require('tapable'); const hook = new AsyncParallelHook(['name', 'age']);
hook.tapAsync('任意字符1', (name, age, callback) => {
// 通过tapAsync方法注册监听;通过callback方法完成监听
console.time(1)
setTimeout(() => {
callback();
},1000)
})
hook.tapAsync('任意字符2', (name, age, callback) => {
setTimeout(() => {
callback();
console.timeEnd(1)
}, 2000)
})
// 通过callAsync方法触发监听;且必须有回调函数
hook.callAsync('lyra', 18, function(e) {
//该函数必须存在
});
2. 自定义创建插件
1. 打包zip插件
const JsZip = require('jszip'); class ZipPlugin {
constructor(options) {
this.options = options;
}
apply(compiler) {
// emit是一个异步串行钩子
compiler.hooks.emit.tapPromise('1', (compilation) => {
const assets = compilation.assets;
const zip = new JsZip();
for(let filename in assets) {
zip.file(filename, assets[filename].source())
}
// nodebuffer是node环境中的二进制形式;blob是浏览器环境
return zip.generateAsync({type: 'nodebuffer'}).then((content) =>{
console.log(this.options.name);
assets[this.options.name] = {
source() {return content},
size() {return content.length} //可以省略
}
return new Promise((resolve, reject) => {
resolve(compilation)
})
})
})
}
} module.exports = ZipPlugin;
在webpack.config.js中使用
const ZipPlugin = require('./plugins/ZipPlugin'); module.exports = {
plugins: [
new ZipPlugin({
name: 'my.zip'
})
]
}
webpack的plugin原理的更多相关文章
- webpack 的编译原理
自从接触了react,vue 这两个框架,都会用到webpack这个打包工具.面试的时候,经常被问到知道webpack的编译原理吗? 可以简单的介绍一下.每每这个时候都被问的哑口无言,平时用的时候挺顺 ...
- 如何编写一个WebPack的插件原理及实践
_ 阅读目录 一:webpack插件的基本原理 二:理解 Compiler对象 和 Compilation 对象 三:插件中常用的API 四:编写插件实战 回到顶部 一:webpack插件的基本原理 ...
- 轻松理解webpack热更新原理
一.前言 - webpack热更新 Hot Module Replacement,简称HMR,无需完全刷新整个页面的同时,更新模块.HMR的好处,在日常开发工作中体会颇深:节省宝贵的开发时间.提升开发 ...
- MyBatis源码分析(2)—— Plugin原理
@(MyBatis)[Plugin] MyBatis源码分析--Plugin原理 Plugin原理 Plugin的实现采用了Java的动态代理,应用了责任链设计模式 InterceptorChain ...
- vim plugin 原理
vim 个性化设置与功能扩展均通过 script 来实现,这种 script 又叫 plugin.plugin 是 vim 的核心与精髓. 最常用的配置文件 vimrc,也是一种 plugin.换句话 ...
- [转] webpack之plugin内部运行机制
简介 webpack作为当前最为流行的模块打包工具,几乎所有的主流前端开发框架(React.Vue等)都会将其作为默认的模块加载和打包工具.通过简单的配置项,使用各种相关的loader和plugin, ...
- Webpack机制、原理简单小结
一.webpack的构成 entry 代表项目的入口 module 开发中,每一个文件可以看作一个module chunk 代码块 loader 模块转化器 plugin 扩展插件,自定义w ...
- webpack之 plugin(插件)
plugin plugin是插件的意思,通常用来对某个现有的架构就行拓展 webpack中的插件,就是对webpack现有功能的各种扩展,比如打包优化,文件压缩等 loader和plugin区别 lo ...
- webpack热替换原理
前期准备: const path = require('path'); const HtmlWebpackPlugin= require('html-webpack-plugin'); const C ...
随机推荐
- Delphi文字转语音TTS【支持选择语音库,播放,暂停,开始,停止,生成语音文件,设置音量,设置语速】
作者QQ:(648437169) 点击下载➨文字转语音TTS [Delphi 文字转语音TTS]调用系统自带的TTS组件,支持XP,vista,win7,win8,win10系统,支持选择语音库,播放 ...
- Django框架(十三)——Auth模块
Auth模块 一.什么是auth模块 Auth模块是Django自带的用户认证模块 Auth模块是Django自带的用户认证模块,可以实现包括用户注册.用户登录.用户认证.注销.修改密码等功能.默认使 ...
- 【1】TOPK最小的K个数(多种方法比较)
(头条) 最小的第K个数也是和这题topK一样的思路 1.全排序 时间复杂度O(nlogn) 2.Partiton思想 时间复杂度O(n) (因为不需要像快排一样对所有的分段都两两Partitio ...
- 通过分析 WPF 的渲染脏区优化渲染性能
原文:通过分析 WPF 的渲染脏区优化渲染性能 本文介绍通过发现渲染脏区来提高渲染性能. 本文内容 脏区 Dirty Region WPF 性能套件 脏区监视 优化脏区重绘 脏区 Dirty Regi ...
- Spring-Cloud之Sleuth链路追踪-8
一.Spring Cloud Sleuth 是Spring Cloud 的一个组件,它的主要功能是在分布式系统中提供服务链路追踪的解决方案. 二.为什么需要Spring Cloud Sleuth? 微 ...
- Spring-Cloud之Config配置中心-7
一.我们前面基本上都是讲解的Spring Cloud Netflix的组件,下面我们会重点说Spring Cloud Config分布式配置中心.为什么需要这个组件来管理配置呢?在分布式应用开发过程中 ...
- 手写MQ框架(三)-客户端实现
一.背景 书接手写MQ框架(二)-服务端实现 ,前面介绍了服务端的实现.但是具体使用框架过程中,用户肯定是以客户端的形式跟服务端打交道的.客户端的好坏直接影响了框架使用的便利性. 虽然框架目前是通过 ...
- vue 属性props定义方法
当子组件接收父组件传过来的值的时候,我们一般有两种方式来接收 不过大家好像都用第二种方式,我只有在不确定数据类型的时候才用第一种方式 第一种: export default { // 不检测类型,全盘 ...
- SAP错误消息调试之七种武器:让所有的错误消息都能被定位
目录 长生剑 - SAPGUI Where Used List 碧玉刀 - ABAP调试器观察点 霸王枪 - ABAP调试器动态断点 多情环 - ABAP代码静态扫描 孔雀翎 - SAT 离别钩 - ...
- ajax请求处理概要
/** *不关心参数传递与参数返回的形式. */ url = ctxPath + '/ccb/xxx '; $.get(url); $.post(url); /** * 常见形式. */ var ur ...