第 27 题:Promise 实现原理(含手写 Promise)

好的,进入 第 32 题:Promise 实现原理(含手写 Promise)

这是 前端顶级高频题,高级前端/资深前端必问。


第 32 题:Promise 的原理是什么?如何手写一个 Promise?(详细版)


一、Promise 的本质是什么?(核心一句话)

Promise 是一个状态机 + 订阅发布模式的异步任务容器。

它有三个状态:

  • pending
  • fulfilled
  • rejected

状态只能:

✔ 从 pending → fulfilled

✔ 从 pending → rejected

✖ 不能从 fulfilled/rejected 回到 pending


二、Promise 做了什么?(面试官要的)

  1. 把异步操作包装成一个对象
  2. 可以通过 then / catch 注册回调
  3. 状态变化后一旦确定,不可逆
  4. 所有 then 回调都是微任务(microtask)
  5. 链式调用支持,返回新的 Promise

三、Promise 的关键实现点(面试必考 5 点)

  1. 状态管理(state)

  2. 成功值 & 失败原因(value,reason)

  3. 回调依赖队列(onFulfilledQueue / onRejectedQueue)

  4. 状态变更后触发回调(发布-订阅)

  5. then 的链式调用要返回新 Promise,处理三种情况:

    • 返回普通值 → resolve
    • 返回 Promise → 等待这个 Promise
    • 抛错 → reject

四、最常见面试要求:手写一个最小 Promise

下面是一个 真正能运行 的、最简版 Promise 实现

面试考这个版本最稳。


五、手写 Promise(简化可运行版)

ini 复制代码
class MyPromise {
  constructor(executor) {
    this.state = 'pending';
    this.value = null;
    this.reason = null;

    this.onFulfilledCallbacks = [];
    this.onRejectedCallbacks = [];

    const resolve = (value) => {
      if (this.state === 'pending') {
        this.state = 'fulfilled';
        this.value = value;
        this.onFulfilledCallbacks.forEach(fn => fn());
      }
    };

    const reject = (reason) => {
      if (this.state === 'pending') {
        this.state = 'rejected';
        this.reason = reason;
        this.onRejectedCallbacks.forEach(fn => fn());
      }
    };

    try {
      executor(resolve, reject);
    } catch (err) {
      reject(err);
    }
  }

  then(onFulfilled, onRejected) {
    // 如果用户没传,就兜底
    onFulfilled = typeof onFulfilled === 'function' ? onFulfilled : (v) => v;
    onRejected = typeof onRejected === 'function' ? onRejected : (e) => { throw e; };

    return new MyPromise((resolve, reject) => {
      if (this.state === 'fulfilled') {
        queueMicrotask(() => {
          try {
            const x = onFulfilled(this.value);
            resolve(x);
          } catch (err) {
            reject(err);
          }
        });
      }

      if (this.state === 'rejected') {
        queueMicrotask(() => {
          try {
            const x = onRejected(this.reason);
            resolve(x);
          } catch (err) {
            reject(err);
          }
        });
      }

      if (this.state === 'pending') {
        this.onFulfilledCallbacks.push(() => {
          queueMicrotask(() => {
            try {
              const x = onFulfilled(this.value);
              resolve(x);
            } catch (err) {
              reject(err);
            }
          });
        });

        this.onRejectedCallbacks.push(() => {
          queueMicrotask(() => {
            try {
              const x = onRejected(this.reason);
              resolve(x);
            } catch (err) {
              reject(err);
            }
          });
        });
      }
    });
  }
}

六、这个 Promise 实现能做什么?

✔ then 链式调用

✔ 异步执行

✔ 回调队列

✔ 状态不可逆

✔ 微任务执行模型

✔ 抛错捕获

基本等同于简化版 ES Promise,面试够用。


七、面试官会继续追问什么?(提前准备)

我给你梳理好了:

1. 如何实现 catch

kotlin 复制代码
catch(errFn) {
  return this.then(null, errFn);
}

2. 如何实现 finally

javascript 复制代码
finally(fn) {
  return this.then(
    v => Promise.resolve(fn()).then(() => v),
    e => Promise.resolve(fn()).then(() => { throw e; })
  );
}

3. 如何处理 then 返回 Promise 本身?

需要 resolvePromise 解析器(完整版才有)

4. then 为什么是微任务?

因为 ES 标准规定:

Promise reaction 必须在当前执行栈结束后、微任务队列中执行。


八、20 秒背诵版(面试必杀)

Promise 是一个状态机(pending → fulfilled/rejected),并使用发布订阅模式来管理回调。then 的回调会被加入微任务队列,状态变更后触发执行。链式调用通过返回新的 Promise 并解析返回结果实现。如果返回值是普通值直接 resolve,如果返回的是 Promise,就等待它完成。


是否进入 第 33 题:async/await 原理(为什么 await 是 Promise 的语法糖)

相关推荐
陈随易11 分钟前
bun将会支持Bun.image,你怎么看?
前端·后端·程序员
jingqingdai328 分钟前
别用正则格式化 HTML!我用 DOM 遍历实现零风险本地格式化,老项目重构效率直接拉满
前端·重构·html
木斯佳30 分钟前
前端八股文面经大全:腾讯前端实习二、三OC面(2026-04-27)·面经深度解析
前端·状态模式
Python私教43 分钟前
如意Agent日志系统重构:从 print() 大海捞针到结构化可观测性栈
java·前端·重构
We་ct1 小时前
LeetCode 97. 交错字符串:动态规划详解
前端·算法·leetcode·typescript·动态规划
Chengbei111 小时前
轻量化 Web 安全日志分析神器 星川智盾日志威胁检测、地理溯源、MITRE ATT&CK 映射,支持 Windows/macOS/Linux
前端·人工智能·安全·web安全·macos·系统安全·安全架构
风流 少年1 小时前
Python Web框架:FastAPI
前端·python·fastapi
GISer_Jing1 小时前
AI时代面试新常态——从“会用工具”到“深挖原理”的跨越
前端·人工智能·ai编程
IT_陈寒2 小时前
React的useEffect把我坑惨了,这些闭包陷阱真要命
前端·人工智能·后端