webpack源码分析——tapable中before和stage如何改变执行顺序

一、Before用法

Before 用法

before 属性的值可以传入一个数组或者字符串,值为注册事件对象时的名称,它可以修改当前事件函数在传入的事件名称对应的函数之前进行执行。

示例

js 复制代码
 let hook = new SyncWaterfallHook(['arg1']);

 hook.tap('tap1', (arg)=> {
	console.log('tap1', arg);
	return '返回值';
 });

 hook.tap({
	name: 'tap2',
	before: 'tap1' // 这里调换了顺序,把 tap2 执行顺序调整到 tap1 之前
 }, (arg)=> {
	console.log('tap2', arg);
	return '返回2';
 })

 hook.tap('tap3', (arg)=> {
	console.log('tap3', arg);
	return '返回3';
 })

 hook.call('yun');


原理描述

tap 函数把注册函数以一下方式放入taps队列

第一次调用tap函数进行注册函数时:

  1. 使用 _insert 函数完成注册函数进入队列
  2. 因为此时taps队列中还没有函数,所以直接当前注册函数通过 this.taps[i] = item; 推入队列中

第二次调用tap函数进行注册函数时:

  1. 使用 _insert 函数完成注册函数进入队列
  2. 如果taps 中有函数,则进入 while 循环
  3. 把taps队列中最后一个函数(下标为0的函数)取出赋值给x变量中并为队列长度延长一个位置后(当前队列长度为2),把x变量赋值给刚才延长的位置上(刚才延长的位置下标为1)
  4. 把要注册的函数放入下标为1的位置

第三、四...次函数注册和第二次注册处理逻辑一致(只是下标有所不同)。

示例:

tap2是要插入的函数

循环前未插入tap2时:taps:['tap1']

循环前未准备插入tap2时:taps:['tap1','tap1']

不再循环完成插入时:taps:['tap1', 'tap2']
tap3是要插入的函数

循环前未插入tap3时:taps:['tap1', 'tap2']

循环前未准备插入tap3时:taps:['tap1','tap2','tap2']

不再循环完成插入时:taps:['tap1','tap2', 'tap3']

当使用 before 进行改变插入顺序时逻辑如下

第一次插入注册函数和上方一致

第二次调用tap函数进行注册函数时:

  1. 使用 _insert 函数完成注册函数进入队列
  2. taps 中已经有函数了,因此进入 while 循环
  3. 把taps队列中最后一个函数取出赋值给x变量中并为队列长度延长一个位置后,把x变量赋值给刚才延长的位置上
  4. 判断是否存在before,如果不存在就没有改变插入顺序
  5. 如果存在,判断当前x.name是不是要被插入的函数
  6. 如果是则continue
  7. 如果循环条件不满足则退出循环
  8. 最后插入被插入函数的前面

第三、四...次函数注册和第二次注册处理逻辑一致。

示例:

tap3是要插入tap1的函数前面

循环前未插入tap3时:taps:['tap1', 'tap2']

循环前未准备插入tap3时:taps:[''tap1','tap2','tap2']

循环前未准备插入tap3时:taps:[''tap1','tap1','tap2']

不再循环完成插入时:taps:['tap3','tap1', 'tap2']

二、stage 用法

stage 用法

stage 类型是数字,数字越大事件回调执行的越晚,支持传入负数,不传时默认为0.

示例

js 复制代码
let hook = new SyncWaterfallHook(['arg1']);

 hook.tap('tap1', (arg)=> {
	console.log('tap1', arg);
	return '返回值';
 });

 hook.tap({
	name: 'tap2',
	stage: 2
 }, (arg)=> {
	console.log('tap2', arg);
	return '返回2';
 })

 hook.tap('tap3', (arg)=> {
	console.log('tap3', arg);
	return '返回3';
 })

 hook.tap({
	name: 'tap4',
	stage: 1
 }, (arg)=> {
	console.log('tap4', arg);
	return '返回4';
 })

 hook.call('yun');


原理描述

原理基本和Before一致。

示例:

已知tap1,tap2 已在taps队列中 taps:['tap1-stage:0', 'tap2-stage:0']

先插入tap3-stage3和tap4-stage2

循环前未插入tap3时:taps:[''tap1-stage:0', ''tap2-stage:0']

循环前未准备插入tap3时:taps:[''tap1-stage:0',''tap2-stage:0',''tap2-stage:0']

不再循环完成插入时:taps:[''tap1-stage:0',''tap2-stage:0',''tap3-stage:3']
循环前未插入tap4时:taps:[''tap1-stage:0',''tap2-stage:0',''tap3-stage:3']

循环前未准备插入tap4时:taps:[''tap1-stage:0',''tap2-stage:0',''tap3-stage:3', ''tap3-stage:3']

循环前未准备插入tap4时:taps:[''tap1-stage:0',''tap2-stage:0',''tap3-stage:0', ''tap3-stage:3']

不再循环完成插入时:taps:[''tap1-stage:0',''tap2-stage:0','"'tap4-stage:2",'tap3-stage:3']

三、源码

js 复制代码
_resetCompilation() {
	this.call = this._call;
	this.callAsync = this._callAsync;
	this.promise = this._promise;
}

_insert(item) {
		this._resetCompilation(); // 重置有关配置
		let before;
		if (typeof item.before === "string") { // 判断传入的 before 字段类型,下方统一为Set类型
			before = new Set([item.before]);
		} else if (Array.isArray(item.before)) {
			before = new Set(item.before);
		}
		let stage = 0;
		if (typeof item.stage === "number") {
			stage = item.stage;
		}
		let i = this.taps.length; // 获取注册的tap函数个数
		while (i > 0) {
			i--;
			const x = this.taps[i];
			this.taps[i + 1] = x; 
			const xStage = x.stage || 0;
			if (before) { // 判断是否可以插入
				if (before.has(x.name)) { // 当前x.name是否是被插入的函数
					before.delete(x.name);
					continue; // 继续循环
				}
				if (before.size > 0) {
					continue; // 继续循环
				}
			}
			if (xStage > stage) {
				continue;
			}
			i++; 
			break;
		}
		this.taps[i] = item;
	}
相关推荐
JELEE.5 小时前
Django登录注册完整代码(图片、邮箱验证、加密)
前端·javascript·后端·python·django·bootstrap·jquery
TeleostNaCl7 小时前
解决 Chrome 无法访问网页但无痕模式下可以访问该网页 的问题
前端·网络·chrome·windows·经验分享
前端大卫8 小时前
为什么 React 中的 key 不能用索引?
前端
你的人类朋友8 小时前
【Node】手动归还主线程控制权:解决 Node.js 阻塞的一个思路
前端·后端·node.js
小李小李不讲道理10 小时前
「Ant Design 组件库探索」五:Tabs组件
前端·react.js·ant design
毕设十刻10 小时前
基于Vue的学分预警系统98k51(程序 + 源码 + 数据库 + 调试部署 + 开发环境配置),配套论文文档字数达万字以上,文末可获取,系统界面展示置于文末
前端·数据库·vue.js
mapbar_front11 小时前
在职场生存中如何做个不好惹的人
前端
牧杉-惊蛰11 小时前
纯flex布局来写瀑布流
前端·javascript·css
一袋米扛几楼9812 小时前
【软件安全】什么是XSS(Cross-Site Scripting,跨站脚本)?
前端·安全·xss
向上的车轮12 小时前
Actix Web适合什么类型的Web应用?可以部署 Java 或 .NET 的应用程序?
java·前端·rust·.net