JS異步:setTimeout包裝為sleep

前言

最近在开发一个浏览器自动化脚本时,遇到了一个看似简单却很重要的异步编程问题:为什么不能直接使用 `setTimeout`,而需要将其包装成一个 `sleep` 函数?这个问题的核心,其实涉及到 JavaScript 的异步编程模型和 Promise 机制。本文将从小白的视角,深入浅出地解释这个概念。

问题场景

在编写自动化脚本时,我们经常需要模拟人类的操作行为,比如:

  1. 输入文本时,需要逐字输入,每个字符之间有延迟

  2. 填写表单时,需要等待一个字段填写完成后再填写下一个

  3. 点击按钮后,需要等待页面响应

最开始的代码可能是这样写的:

复制代码
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));这行代码做了什么?

  1. 创建一个新的 Promise

  2. 在 Promise 内部调用 `setTimeout`

  3. 当 `setTimeout` 的回调执行时,调用 `resolve`(这里的 `r` 就是 `resolve`)

  4. 返回这个 Promise,这样就能用 `await` 等待了

简单来说:**`sleep` 将不返回 Promise 的 `setTimeout`,包装成了返回 Promise 的函数**。

对比:setTimeout vs sleep

|----------|--------------|-------------|
| 特性 | setTimeout | sleep (包装后) |
| 返回值 | 数字(timer ID) | Promise |
| 能否 await | 不能 | 能 |
| 是否阻塞执行 | 不阻塞 | 阻塞(在当前函数内) |
| 适用场景 | 延迟执行独立任务 | 需要顺序执行的异步操作 |

相关推荐
秋邱12 分钟前
用 Python 写出 C++ 的性能?用CANN中PyPTO 算子开发硬核上手指南
开发语言·c++·python
wenzhangli733 分钟前
ooderA2UI BridgeCode 深度解析:从设计原理到 Trae Solo Skill 实践
java·开发语言·人工智能·开源
晚烛1 小时前
CANN + 物理信息神经网络(PINNs):求解偏微分方程的新范式
javascript·人工智能·flutter·html·零售
灵感菇_1 小时前
Java 锁机制全面解析
java·开发语言
wazmlp0018873691 小时前
python第三次作业
开发语言·python
娇娇乔木1 小时前
模块十一--接口/抽象方法/多态--尚硅谷Javase笔记总结
java·开发语言
明月醉窗台1 小时前
qt使用笔记六之 Qt Creator、Qt Widgets、Qt Quick 详细解析
开发语言·笔记·qt
wangjialelele1 小时前
平衡二叉搜索树:AVL树和红黑树
java·c语言·开发语言·数据结构·c++·算法·深度优先
lili-felicity1 小时前
CANN性能调优与实战问题排查:从基础优化到排障工具落地
开发语言·人工智能
独自破碎E1 小时前
【BISHI15】小红的夹吃棋
android·java·开发语言