面试手写 Promise:链式 + 静态方法全实现

ES5 语法 实现一个 符合 Promise/A+ 规范 的完整 Promise,附 Promise.all/race/finally 等静态方法,面试手写无压力。


一、手写核心 Promise(符合 A+)

1. 结构

js 复制代码
const PENDING = 'pending';
const FULFILLED = 'fulfilled';
const REJECTED = 'rejected';

function MyPromise(executor) {
  this.state = PENDING;
  this.value = undefined;
  this.reason = undefined;
  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);
  }
}

2. then 方法(链式核心)

js 复制代码
MyPromise.prototype.then = function (onFulfilled, onRejected) {
  onFulfilled = typeof onFulfilled === 'function' ? onFulfilled : val => val;
  onRejected = typeof onRejected === 'function' ? onRejected : err => { throw err; };

  return new MyPromise((resolve, reject) => {
    const handle = () => {
      try {
        const result = this.state === FULFILLED
          ? onFulfilled(this.value)
          : onRejected(this.reason);
        resolvePromise(result, resolve, reject);
      } catch (err) {
        reject(err);
      }
    };

    if (this.state === PENDING) {
      this.onFulfilledCallbacks.push(handle);
      this.onRejectedCallbacks.push(handle);
    } else {
      setTimeout(handle, 0);
    }
  });
};

function resolvePromise(result, resolve, reject) {
  if (result instanceof MyPromise) {
    result.then(resolve, reject);
  } else {
    resolve(result);
  }
}

二、静态方法全家桶

1. Promise.resolve / reject

js 复制代码
MyPromise.resolve = value => new MyPromise(resolve => resolve(value));
MyPromise.reject = reason => new MyPromise((_, reject) => reject(reason));

2. Promise.all(并发等待)

js 复制代码
MyPromise.all = function (promises) {
  return new MyPromise((resolve, reject) => {
    let count = 0;
    const results = [];
    promises.forEach((p, i) => {
      MyPromise.resolve(p).then(
        val => {
          results[i] = val;
          if (++count === promises.length) resolve(results);
        },
        reject
      );
    });
  });
};

3. Promise.race(谁先完成用谁)

js 复制代码
MyPromise.race = function (promises) {
  return new MyPromise((resolve, reject) => {
    promises.forEach(p => MyPromise.resolve(p).then(resolve, reject));
  });
};

4. Promise.finally(无论成功失败都执行)

js 复制代码
MyPromise.prototype.finally = function (callback) {
  return this.then(
    val => MyPromise.resolve(callback()).then(() => val),
    err => MyPromise.resolve(callback()).then(() => { throw err; })
  );
};

三、完整测试用例

js 复制代码
// 1. 基本链式
new MyPromise(resolve => resolve(1))
  .then(val => val + 1)
  .then(val => console.log(val)); // 2

// 2. 并发
MyPromise.all([
  new MyPromise(resolve => setTimeout(() => resolve('a'), 100)),
  new MyPromise(resolve => setTimeout(() => resolve('b'), 200))
]).then(console.log); // ['a', 'b']
相关推荐
IT_陈寒13 分钟前
Python 3.12 新特性实战:5个让你的代码效率提升50%的技巧!🔥
前端·人工智能·后端
Apifox15 分钟前
Apifox 8 月更新|新增测试用例、支持自定义请求示例代码、提升导入/导出 OpenAPI/Swagger 数据的兼容性
前端·后端·测试
weixin_5412999417 分钟前
VSCode: 从插件安装到配置,如何实现 Ctrl+S 保存时,完全按照 .eslintrc.js 中的 ESLint 规则自动格式化代码
javascript·ide·vscode
yw00yw17 分钟前
常见的设计模式
开发语言·javascript·设计模式
coding随想23 分钟前
最后的挽留:深入浅出HTML5 beforeunload事件
前端
叶浩成52026 分钟前
WebSocket实时通信系统——js技能提升
javascript·websocket·网络协议
亚里士多德芙35 分钟前
记录:离线包实现桥接
前端
去伪存真1 小时前
用的好好的vue.config.js代理,突然报308, 怎么回事?🤔
前端
在未来等你1 小时前
RabbitMQ面试精讲 Day 29:版本升级与平滑迁移
中间件·面试·消息队列·rabbitmq
搞个锤子哟1 小时前
el-select使用filter-method实现自定义过滤
前端