JavaScript异步编程:Promise与async/await

1.什么是Promise

Promise,承诺,在 JavaScript 中,它是一种异步编程的解决方案,Promise 是一个 表示异步操作最终完成或失败的对象,它提供了一种标准化的方式来处理异步结果。你可以把它想象成一个"容器",里面包裹着一个未来才会知道结果的操作(比如网络请求、定时器、文件读取等)。

Promise 的三种状态

  • Pending(进行中):初始状态,操作尚未完成。
  • Fulfilled(已成功):操作成功完成,Promise 返回结果值。
  • Rejected(已失败):操作失败,Promise 返回错误原因。

Promise 构造函数接收一个函数(称为 执行器函数,executor ),这个执行器函数会被 立即同步调用 (也就是 new Promise 时立刻执行)。

执行器函数的两个参数:

  • resolve(value)

    将 Promise 的状态从 pending 变为 fulfilled(已完成),并传递一个值给后续的 .then()

    value参数规则

    参数类型 行为
    普通值 直接作为成功值传递,例如 resolve(42)resolve("data")
    Promise 对象 当前 Promise 会"跟随"这个 Promise 的状态(成为它的镜像)。
    无参数 等同于 resolve(undefined)
  • reject(value)

    调用它表示 异步操作失败 ,并将 Promise 状态从 pending 变为 rejected(失败)。

    可以传递一个错误原因(如 reject(new Error("超时"))),这个错误会被.then()的第二个参数.catch() 捕获(向下传递)。通常推荐传递 Error 对象(如 reject(new Error("失败"))),但可以是任意值(如字符串、数字等)。

2.Promise.then()的作用

  • 接收上游 Promise 的结果。
  • 返回新 Promise 供下游使用。
  • 隐式 处理 resolve/reject,开发者只需关心业务逻辑。
javascript 复制代码
// 伪代码:.then() 的底层实现
function then(onFulfilled) {
  return new Promise((resolve, reject) => {
    const handleResult = (value) => {
      try {
        const result = onFulfilled(value); // 执行你的回调,value指当前promise的result
        // 自动处理返回值!
        if (result instanceof Promise) {
          result.then(resolve, reject); // 跟随返回的 Promise
        } else {
          resolve(result); // 普通值直接 resolve
        }
      } catch (error) {
        reject(error); // 捕获错误并 reject
      }
    };
    this.addHandler(handleResult); // 监听当前 Promise 状态
  });
}

3.async/await语法糖

Promise 构造函数接收一个函数(称为 执行器函数,executor),执行器函数的 函数体*就是定义 "未来要发生的动作" ,通常是一个异步操作(如 setTimeoutfetchfs.readFile 等)。

当这个 未来动作完成时 ,你需要 手动调用 resolvereject 来改变 Promise 的状态。每次手动调用resolve和reject会非常繁琐,所以async/await语法糖应运而生

对比示例:用 Promise 和 async/await 实现同一逻辑

场景:先请求用户数据,再请求用户订单,最后合并结果。

手动控制 Promise 的写法(繁琐)

javascript 复制代码
function getUserData(userId) {
  return new Promise((resolve, reject) => {
    fetch(`/users/${userId}`)
      .then(response => {
        if (!response.ok) reject(new Error("用户数据请求失败"));
        return response.json();
      })
      .then(data => resolve(data)) // 手动 resolve
      .catch(error => reject(error)); // 手动 reject
  });
}

function getOrders(userId) {
  return new Promise((resolve, reject) => {
    fetch(`/orders/${userId}`)
      .then(response => {
        if (!response.ok) reject(new Error("订单请求失败"));
        return response.json();
      })
      .then(data => resolve(data)) // 手动 resolve
      .catch(error => reject(error)); // 手动 reject
  });
}

// 链式调用(仍需手动处理嵌套)
getUserData(123)
  .then(user => {
    getOrders(user.id)
      .then(orders => console.log({ user, orders }))
      .catch(error => console.error("订单错误:", error));
  })
  .catch(error => console.error("用户错误:", error));

需要反复手动调用 resolve/reject

多层嵌套时容易陷入"回调地狱"(虽然比传统回调好,但依然不够直观)。

用 async/await 的写法(简洁)

javascript 复制代码
async function fetchUserAndOrders(userId) {
  try {
    const user = await fetch(`/users/${userId}`).then(res => res.json()); // 自动解析 Promise
    const orders = await fetch(`/orders/${user.id}`).then(res => res.json());
    console.log({ user, orders });
  } catch (error) {
    console.error("请求失败:", error); // 统一捕获错误
  }
}

fetchUserAndOrders(123);
相关推荐
云水一下4 小时前
Vue.js从零到精通系列(三):组件化基础——Props、Emits、插槽与生命周期
前端·javascript·vue.js
小糯米6014 小时前
JavaScript表达式与运算符
开发语言·javascript·ecmascript
体验家5 小时前
体验家 XMPlus 网页端问卷 SDK 技术解析:用几行 JavaScript 实现精准场景触发与防打扰机制
开发语言·前端·javascript
VidDown5 小时前
VidDown 工具站:视频分辨率技术
javascript·网络·编辑器·音视频·视频编解码·视频
小鹿软件办公5 小时前
倒计时开启:Chromium 宣布几周内将全面切断 MV2 扩展支持
开发语言·javascript·ublock origin
Csvn6 小时前
TypeScript:你以为安全的 `JSON.parse` 其实是颗雷 — 运行时类型安全实战
前端·javascript
触底反弹6 小时前
从 JS 引擎执行原理理解数据类型:栈内存、堆内存与作用域
javascript·数据结构·面试
橘子星6 小时前
深入理解线性数据结构:栈、队列与链表
前端·javascript
Larcher6 小时前
JS 数据类型的八重人格与内存真相
前端·javascript
Maimai108086 小时前
Web3 前端实时通信如何落地:从 SSE 订阅到行情、订单与账户状态更新
前端·javascript·react.js·前端框架·web3·状态模式