手写Promise最终版本

手写 Promise 全实现

目标:完整实现一个符合 Promise/A+ 行为模型的 Promise ,覆盖 then 链式调用、值穿透、异常捕获、Promise.resolve / reject / all / race`,并严格使用 微任务(queueMicrotask)


一、整体设计思路

Promise 的三大核心要素

  1. 状态管理

    • pending
    • fulfilled
    • rejected
  2. 结果存储

    • 成功值 / 失败原因只能设置一次
  3. 回调队列

    • pending 状态下,then 注册的回调需要被缓存
    • 状态确定后统一执行

二、Promise 构造函数实现

kotlin 复制代码
function Promise(executor) {
  this.PromiseState = "pending"
  this.PromiseResult = null
  this.callBacks = []

  const self = this

  function resolve(data) {
    if (self.PromiseState !== "pending") return

    // 处理 resolve(Promise)
    if (data instanceof Promise) {
      data.then(resolve, reject)
      return
    }

    self.PromiseState = "fulfilled"
    self.PromiseResult = data

    self.callBacks.forEach(cb => {
      if (cb.onResolve) {
        queueMicrotask(() => cb.onResolve(data))
      }
    })
  }

  function reject(data) {
    if (self.PromiseState !== "pending") return

    // 处理 reject(Promise)
    if (data instanceof Promise) {
      data.then(resolve, reject)
      return
    }

    self.PromiseState = "rejected"
    self.PromiseResult = data

    self.callBacks.forEach(cb => {
      if (cb.onReject) {
        queueMicrotask(() => cb.onReject(data))
      }
    })
  }

  try {
    executor(resolve, reject)
  } catch (error) {
    reject(error)
  }
}

关键点解析

  • 状态只能从 pending → fulfilled / rejected
  • resolve / reject 内部必须防止重复执行
  • executor 同步异常需要被 reject 捕获
  • 所有 then 回调必须异步执行(微任务)

三、then 方法实现(核心)

javascript 复制代码
Promise.prototype.then = function (onResolve, onReject) {
  const self = this

  // 值穿透 & 异常穿透
  if (typeof onResolve !== "function") {
    onResolve = result => result
  }
  if (typeof onReject !== "function") {
    onReject = reason => { throw reason }
  }

  return new Promise((resolve, reject) => {
    function callFunction(fn) {
      try {
        const result = fn(self.PromiseResult)
        if (result instanceof Promise) {
          result.then(resolve, reject)
        } else {
          resolve(result)
        }
      } catch (error) {
        reject(error)
      }
    }

    if (self.PromiseState === "fulfilled") {
      queueMicrotask(() => callFunction(onResolve))
    }

    if (self.PromiseState === "rejected") {
      queueMicrotask(() => callFunction(onReject))
    }

    if (self.PromiseState === "pending") {
      self.callBacks.push({
        onResolve: () => callFunction(onResolve),
        onReject: () => callFunction(onReject)
      })
    }
  })
}

then 的核心规则

  1. then 必须返回一个新的 Promise

  2. 回调返回值:

    • 普通值 → resolve
    • Promise → 状态跟随
    • throw → reject
  3. 支持链式调用 & 错误冒泡


四、Promise.resolve / Promise.reject

javascript 复制代码
Promise.resolve = function (data) {
  return new Promise((resolve, reject) => {
    if (data instanceof Promise) {
      data.then(resolve, reject)
    } else {
      resolve(data)
    }
  })
}

Promise.reject = function (data) {
  return new Promise((resolve, reject) => {
    reject(data)
  })
}

五、Promise.race 实现

javascript 复制代码
Promise.race = function (promises) {
  return new Promise((resolve, reject) => {
    for (let i = 0; i < promises.length; i++) {
      promises[i].then(resolve, reject)
    }
  })
}

行为特点

  • 谁先改变状态,用谁的结果
  • 不关心其它 Promise 状态

六、Promise.all 实现

ini 复制代码
Promise.all = function (promises) {
  let count = 0
  const result = []

  return new Promise((resolve, reject) => {
    for (let i = 0; i < promises.length; i++) {
      promises[i].then(value => {
        result[i] = value
        count++
        if (count === promises.length) {
          resolve(result)
        }
      }, reject)
    }
  })
}

行为特点

  • 顺序与输入一致
  • 任何一个 reject,整体 reject

七、完整能力覆盖检查表

  • 状态不可逆
  • executor 异常捕获
  • then 链式调用
  • 值穿透 / 错误穿透
  • 微任务调度
  • Promise.resolve / reject
  • Promise.all / race

八、总结

这版 Promise:

  • 面试可手写级别
  • Promise/A+ 思维完整实现
  • 已具备真实 Promise 的 90% 行为特征

📌 一句话总结

真正理解 Promise,不是会用,而是能写出来。

相关推荐
visnix2 小时前
AI大模型-LLM原理剖析到训练微调实战(第二部分:大模型核心原理与Transformer架构)
前端·llm
老妪力虽衰2 小时前
零基础的小白也能通过AI搭建自己的网页应用
前端
该用户已不存在2 小时前
Node.js后端开发必不可少的7个核心库
javascript·后端·node.js
踏浪无痕2 小时前
计算机算钱为什么会算错?怎么解决?
后端·算法·面试
褪色的笔记簿2 小时前
在 Vue 项目里管理弹窗组件:用 ref 还是用 props?
前端·vue.js
Danny_FD2 小时前
使用Taro实现微信小程序仪表盘:使用canvas实现仪表盘(有仪表盘背景,也可以用于Web等)
前端·taro·canvas
掘金安东尼2 小时前
VSCode V1.107 发布(2025 年 11 月)
前端·visual studio code
一只小阿乐2 小时前
前端vue3 web端中实现拖拽功能实现列表排序
前端·vue.js·elementui·vue3·前端拖拽
AAA阿giao2 小时前
从“操纵绳子“到“指挥木偶“:Vue3 Composition API 如何彻底改变前端开发范式
开发语言·前端·javascript·vue.js·前端框架·vue3·compositionapi