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;
	}
相关推荐
masa01023 分钟前
JavaScript--JavaScript基础
开发语言·javascript
让开,我要吃人了2 小时前
HarmonyOS开发实战(5.0)实现二楼上划进入首页效果详解
前端·华为·程序员·移动开发·harmonyos·鸿蒙·鸿蒙系统
everyStudy3 小时前
前端五种排序
前端·算法·排序算法
甜兒.4 小时前
鸿蒙小技巧
前端·华为·typescript·harmonyos
她似晚风般温柔7897 小时前
Uniapp + Vue3 + Vite +Uview + Pinia 分商家实现购物车功能(最新附源码保姆级)
开发语言·javascript·uni-app
Jiaberrr8 小时前
前端实战:使用JS和Canvas实现运算图形验证码(uniapp、微信小程序同样可用)
前端·javascript·vue.js·微信小程序·uni-app
everyStudy8 小时前
JS中判断字符串中是否包含指定字符
开发语言·前端·javascript
城南云小白8 小时前
web基础+http协议+httpd详细配置
前端·网络协议·http
前端小趴菜、8 小时前
Web Worker 简单使用
前端
web_learning_3218 小时前
信息收集常用指令
前端·搜索引擎