前言
最近在开发一个浏览器自动化脚本时,遇到了一个看似简单却很重要的异步编程问题:为什么不能直接使用 `setTimeout`,而需要将其包装成一个 `sleep` 函数?这个问题的核心,其实涉及到 JavaScript 的异步编程模型和 Promise 机制。本文将从小白的视角,深入浅出地解释这个概念。
问题场景
在编写自动化脚本时,我们经常需要模拟人类的操作行为,比如:
-
输入文本时,需要逐字输入,每个字符之间有延迟
-
填写表单时,需要等待一个字段填写完成后再填写下一个
-
点击按钮后,需要等待页面响应
最开始的代码可能是这样写的:
function fillForm() {
el1.value = '姓名'; // 立即执行
setTimeout(() => {
el2.value = '电话'; // 1秒后执行
}, 1000);
el3.value = '地址'; // 立即执行(不会等待上面的 setTimeout)
}
问题出现了:虽然 `setTimeout` 会在 1 秒后执行 `el2.value = '电话'`,但是 `el3.value = '地址'` 不会等待,而是立即执行。这导致三个字段几乎同时被填充,失去了模拟人类操作的效果。
setTimeout 的本质
`setTimeout` 是 JavaScript 提供的一个原生 API,它的作用是:**在指定时间后执行回调函数,但不会阻塞后续代码的执行**。
const timerId = setTimeout(() => {
console.log('1秒后执行');
}, 1000);
console.log('立即执行');
console.log('timerId:', timerId); // 输出:timerId: 1(一个数字)
// 执行顺序:
// 1. "立即执行"
// 2. "timerId: 1"
// 3. (1秒后) "1秒后执行"关键点在于:**`setTimeout` 返回的是一个定时器 ID(数字),而不是 Promise**。这意味着它无法与 `await` 配合使用,因为 `await` 只能等待 Promise 对象。
Promise 是什么?
Promise 是 JavaScript 中处理异步操作的一种机制。可以把它想象成一张"欠条",表示"将来某个时刻会给你结果"。
Promise 有三种状态:
-
**pending(等待中)**:操作还在进行,结果未知
-
**fulfilled(已完成)**:操作成功完成
-
**rejected(已拒绝)**:操作失败
// 创建一个 Promise
const myPromise = new Promise((resolve, reject) => {
// resolve = 成功时调用
// reject = 失败时调用setTimeout(() => { resolve('操作成功!'); // 1秒后调用 resolve,表示成功 }, 1000);});
await 的作用
`await` 是配合 `async` 函数使用的关键字,它的作用是:**等待 Promise 完成,然后继续执行后续代码**。
// 正确示例:使用 await
async function fillForm() {
el1.value = '姓名';
await sleep(1000); // 等待 1 秒,代码在这里暂停
el2.value = '电话'; // 1秒后才执行
el3.value = '地址'; // 在 el2 之后执行
}
关键在于:**`await` 会"阻塞"代码执行**,直到 Promise 完成。这里的"阻塞"是指在当前函数内暂停,不会影响页面的其他操作。
sleep 函数的实现
理解了 Promise 和 await 后,我们就能理解为什么要包装 `setTimeout` 了:
const sleep = ms => new Promise(r => setTimeout(r, ms));这行代码做了什么?
-
创建一个新的 Promise
-
在 Promise 内部调用 `setTimeout`
-
当 `setTimeout` 的回调执行时,调用 `resolve`(这里的 `r` 就是 `resolve`)
-
返回这个 Promise,这样就能用 `await` 等待了
简单来说:**`sleep` 将不返回 Promise 的 `setTimeout`,包装成了返回 Promise 的函数**。
对比:setTimeout vs sleep
|----------|--------------|-------------|
| 特性 | setTimeout | sleep (包装后) |
| 返回值 | 数字(timer ID) | Promise |
| 能否 await | 不能 | 能 |
| 是否阻塞执行 | 不阻塞 | 阻塞(在当前函数内) |
| 适用场景 | 延迟执行独立任务 | 需要顺序执行的异步操作 |