第 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 的语法糖)

相关推荐
namexingyun8 分钟前
开源前端生态如何成为 AI UI 生成的“燃料“:shadcn/ui、Tailwind CSS、Storybook 技术价值全解剖
java·前端·人工智能·python·ui·开源·ai编程
Zyed11 分钟前
[STM32]Day15读写FLASH+读取ID
前端·stm32·性能优化
AI人工智能+电脑小能手34 分钟前
【大白话说Java面试题 第110题】【并发篇】第10题:CAS 存在哪些问题?
java·开发语言·面试
秋91 小时前
Python工程师面试常问提问和回答(AI工程化方向 · 2026版)
人工智能·python·面试
jvxiao1 小时前
你真的懂作用域吗?从编译原理角度深度 JS 的作用域
前端·javascript
Darling噜啦啦1 小时前
二叉树与递归算法实战:从树结构到 LeetCode 爬楼梯,一文吃透前端数据结构与递归思维
前端·javascript·数据结构
星栈1 小时前
Rust + Makepad 应用怎么打包发布:Windows、macOS、Linux 全平台交付
前端·rust
Aolith1 小时前
React 路由守卫:我用一个组件替代了 Vue 的 beforeEach
前端·react.js
Daybreak1 小时前
从 PDD、DDD、SDD 到 TDD:我是如何用一套 Agent 工程方法论推进 My-Notion 的
前端
西安邮电大学2 小时前
贪心算法详细讲解
java·后端·其他·算法·面试