概述
工程化配置webpack中,在配置loader的时候,会看到能够写链式调用的写法,如下:
js
config.module
.rule('vue') // 创建一个名为 'vue' 的规则
.test(/\.vue$/) // 规则的条件:匹配 .vue 文件
.use('vue-loader') // 使用一个名为 'vue-loader' 的 loader
.loader('vue-loader') // 指定 loader 的包名
.end()
.end()
.rule('css') // 创建另一个名为 'css' 的规则
.test(/\.css$/)
.use('style-loader')
.loader('style-loader')
.end()
.use('css-loader')
.loader('css-loader');
这里实现类似链式调用的一个函数功能:有时候可能项目上需要插入多个任务,之间是链式执行的情况,通过链式调用来定义一个任务队列,这个队列可以混合立即执行的同步任务 和需要等待的异步任务(如定时器/异步任务) 。特别的是,它提供了一个 .waitFirst()
方法,可以将一个异步等待任务插入到整个队列的最前面 执行。最后调用 .execute()
来按顺序执行整个队列。
实现效果
任务:
js
// 任务
createTask('A')
.wait(2000).doSomething("第一件事")
.wait(3000).doSomething("第二件事")
.waitFirst(1000)
.execute();
输出:
js
Waited first for 1000 ms
Task A start
Waited for 2000 ms
Doing something in task 第一件事
Waited for 3000 ms
Doing something in task 第二件事
实现
js
// 定义一个创建任务的函数,接受一个任务名称
function createTask(name) {
// 初始化一个空数组,用于存储后续由各种方法添加的任务函数
const taskList = []
// 1. 首先,将一个"任务开始"的同步任务加入队列
// 这是一个立即执行的同步任务,用于标记任务开始
taskList.push(() => {
console.log(`Task ${name} start`);
})
// 定义 wait 方法,它接受一个毫秒数 ms
function wait(ms) {
// 向任务队列尾部推送一个新的异步任务函数
taskList.push(() => {
// 这个任务返回一个 Promise,使其成为一个可等待的异步任务
return new Promise((resolve) => {
// 使用 setTimeout 实现等待
setTimeout(() => {
console.log(`Waited for ${ms} ms`);
resolve(); // 等待完成后,解析 Promise,让队列继续执行下一个任务
}, ms);
});
})
// 返回 this(即 createTask 返回的对象),实现链式调用
return this
}
// 定义 doSomething 方法,它接受一个消息 msg
function doSomething(msg) {
// 向任务队列尾部推送一个新的同步任务函数
taskList.push(() => {
console.log(`Doing something in task ${msg}`);
})
// 返回 this,实现链式调用
return this
}
// 定义 waitFirst 方法,它接受一个毫秒数 ms
// 这是关键方法,它与 wait 不同,将任务插入队列头部
function waitFirst(ms) {
// 使用 unshift 而不是 push,将任务插入到数组的最开头
taskList.unshift(() => {
return new Promise((resolve) => {
setTimeout(() => {
console.log(`Waited first for ${ms} ms`);
resolve();
}, ms);
});
})
// 返回 this,实现链式调用
return this
}
// 定义 execute 方法,用于开始按顺序执行任务队列
function execute() {
// 使用立即执行的异步函数表达式 (IIFE) 来执行异步遍历
(async function () {
// 使用 for...of 循环按顺序遍历任务队列
for (const task of taskList) {
// 使用 await 等待当前任务执行完成
// 如果任务是同步的(如 doSomething),会立即执行
// 如果任务是异步的并返回 Promise(如 wait, waitFirst),会等待其 resolve 后再继续
await task()
}
})() // 立即调用这个异步函数
// 仍然返回 this,虽然通常 execute 是链式调用的终点,但保持了API的一致性
return this
}
// 返回一个对象,暴露出的方法可以供外部链式调用
// 注意:这里返回的方法闭包引用了内部的 taskList 数组,这就是它们能操作同一个队列的原因
return {
wait,
doSomething,
waitFirst,
execute
}
}
// 使用示例:创建一个名为 'A' 的任务
createTask('A')
// 链式调用:定义等待2秒后做第一件事
.wait(2000).doSomething("第一件事")
// 链式调用:定义再等待3秒后做第二件事
.wait(3000).doSomething("第二件事")
// 关键操作:将"首先等待1秒"这个任务插入到整个队列的最前面
.waitFirst(1000)
// 开始执行整个定义好的任务队列
.execute();