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 | 不能 | 能 |
| 是否阻塞执行 | 不阻塞 | 阻塞(在当前函数内) |
| 适用场景 | 延迟执行独立任务 | 需要顺序执行的异步操作 |

相关推荐
编织幻境的妖1 小时前
Python垃圾回收机制详解
开发语言·python
遇印记1 小时前
javaOCA考点(基础)
java·开发语言·青少年编程
学困昇1 小时前
Linux基础开发工具(下):调试器gdb/cgdb的使用详解
linux·运维·服务器·开发语言·c++
Junsen1 小时前
使用 Supabase 实现轻量埋点监控
前端·javascript
Java&Develop1 小时前
html写一个象棋游戏
javascript·游戏·html
willxiao2 小时前
js 单例模式 6 种实现方式
javascript·设计模式
biter down2 小时前
C++ 组合与继承:从设计本质到实战,吃透高内聚低耦合
开发语言·c++
灰灰勇闯IT2 小时前
C语言实战:字符串元音字母提取器的实现与优化
c语言·开发语言
fantasy5_52 小时前
C++11 核心特性实战博客
java·开发语言·c++