手写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,不是会用,而是能写出来。

相关推荐
智航GIS9 小时前
10.4 Selenium:Web 自动化测试框架
前端·python·selenium·测试工具
前端工作日常10 小时前
我学习到的A2UI概念
前端
徐同保10 小时前
为什么修改 .gitignore 后还能提交
前端
一只小bit10 小时前
Qt 常用控件详解:按钮类 / 显示类 / 输入类属性、信号与实战示例
前端·c++·qt·gui
Mr -老鬼11 小时前
前端静态路由与动态路由:全维度总结与实践指南
前端
Nan_Shu_61411 小时前
学习: Threejs (1)
javascript·学习
颜酱11 小时前
前端必备动态规划的10道经典题目
前端·后端·算法
Chan1611 小时前
【 Java八股文面试 | JavaSE篇 】
java·jvm·spring boot·面试·java-ee·八股
wen__xvn11 小时前
代码随想录算法训练营DAY10第五章 栈与队列part01
java·前端·算法
Van_Moonlight12 小时前
RN for OpenHarmony 实战 TodoList 项目:加载状态 Loading
javascript·开源·harmonyos