手写Promise-构造函数

在上篇手写Promise-什么是Promise中,我们已经了解了Promise的基本概念和用法。本文将开始用ES5的语法手写Promise的实现,首先从构造函数部分开始讲解。

基础结构搭建

在上篇手写Promise-什么是Promise已经了解到Promise是一个构造函数,接收一个executor参数并立即执行。executor接收两个参数:resolvereject,都是函数类型。可以写成如下代码:

js 复制代码
(function () {
  // 开启严格模式
  "use strict";
  function Promise(executor) {
    // 存储this,方便后续使用
    var self = this;
    // resolve函数接收一个操作成功后的值
    var resolve = function (value) {};
    // reject函数接收一个操作失败的原因
    var reject = function (reason) {};
    executor(resolve, reject);
    // 暴露给浏览器
    if (typeof window !== "undefined") {
      window.Promise = Promise;
    }
    // 暴露给Node.js环境
    if (typeof module === "object" && typeof module.exports === "object") {
      module.exports = Promise;
    }
  }
})();

状态管理

Promise需要记录当前操作的状态(pendingfulfilledrejected)以及操作结果。状态一旦改变就不可逆转 ,这是Promise的重要特性。还需要存储操作成功时的值或操作失败的原因。为result

js 复制代码
(function () {
  "use strict";
  function Promise(executor) {
    // 存储this,方便后续使用
    var self = this;
    // 状态常量
    var PENDING = "pending";
    var FULFILLED = "fulfilled";
    var REJECTED = "rejected";
    self.state = PENDING; // 初始状态
    self.result = void 0;
    // resolve函数接收一个操作成功后的值
    var resolve = function (value) {};
    // reject函数接收一个操作失败的原因
    var reject = function (reason) {};
    executor(resolve, reject);
  }
})();

异常处理

executor可能出现异常,需要捕获并将状态改为rejected。所以捕获到需要执行reject函数。

js 复制代码
(function () {
  // 开启严格模式
  "use strict";
  function Promise(executor) {
    /* 代码省略 */
    // resolve函数接收一个操作成功后的值
    var resolve = function (value) {};
    // reject函数接收一个操作失败的原因
    var reject = function (reason) {};
    try {
      executor(resolve, reject);
    } catch(error) {
      reject(error);
    }
  }
})();

特别注意: try-catch这里无法捕获异步操作的异常。例如以下代码:

js 复制代码
var p = new Promise((resolve) => {
    setTimeout(() => {
        throw new Error(111)
    })
})

参数合法性校验

为了代码健壮性,对参数executor进行校验。

js 复制代码
(function () {
  // 开启严格模式
  "use strict";
  function Promise(executor) {
    // 存储this,方便后续使用
    var self = this;
    /* 代码省略 */
    if (typeof executor !== "function") {
      // 不是函数但是一个对象的报错
      if (typeof executor === "object" && executor !== null) {
        throw new TypeError(
          "TypeError: Promise resolver #<Object> is not a function"
        );
      }
      // 不是一个函数的报错
      throw new TypeError(`Promise resolver ${executor} is not a function`);
    }
    // 构造函数调用检查 - 必须通过new方式调用
    if (!(self instanceof Promise)) {
      throw new TypeError(
        `Promise constructor cannot be invoked without 'new'`
      );
    }
    /* 其余代码 */
  }
})();

初步实现resolve和reject函数

resolve和reject函数分别是成功和失败时调用的。同时当状态是pending时进行修改成对应的状态。

js 复制代码
var resolve = function (value) {
  // 防止后续继续调用resolve或reject
  if (self.state !== PENDING) return;
  self.state = FULFILLED;
  self.result = value;
};
var reject = function (reason) {
  // 防止后续继续调用resolve或reject
  if (self.state !== PENDING) return;
  self.state = REJECTED;
  self.result = reason;
};

小结

本文实现的Promise构造函数具有以下特点:

✅ 已实现功能

  1. 基本结构:完整的构造函数框架
  2. 状态管理:三种状态(pending/fulfilled/rejected)的管理
  3. 状态不可逆:确保状态只能改变一次
  4. 异常处理:同步错误的捕获机制
  5. 参数校验:严格的类型和调用方式检查
  6. 环境适配:浏览器和Node.js环境支持

🎯 下一阶段目标

下一篇文章将实现Promise的核心------then方法,包括:

  • 回调函数的异步执行
  • 链式调用支持
  • 值穿透机制 希望对你有些许帮助!┏(^0^)┛
相关推荐
踩着两条虫1 天前
VTJ.PRO 核心架构全公开!从设计稿到代码,揭秘AI智能体如何“听懂人话”
前端·vue.js·ai编程
jzlhll1231 天前
kotlin Flow first() last()总结
开发语言·前端·kotlin
用头发抵命1 天前
Vue 3 中优雅地集成 Video.js 播放器:从组件封装到功能定制
开发语言·javascript·ecmascript
蓝冰凌1 天前
Vue 3 中 defineExpose 的行为【defineExpose暴露ref变量】详解:自动解包、响应性与实际使用
前端·javascript·vue.js
奔跑的呱呱牛1 天前
generate-route-vue基于文件系统的 Vue Router 动态路由生成工具
前端·javascript·vue.js
柳杉1 天前
从动漫水面到赛博飞船:这位开发者的Three.js作品太惊艳了
前端·javascript·数据可视化
Greg_Zhong1 天前
前端基础知识实践总结,每日更新一点...
前端·前端基础·每日学习归类
We་ct1 天前
LeetCode 148. 排序链表:归并排序详解
前端·数据结构·算法·leetcode·链表·typescript·排序算法
TON_G-T1 天前
day.js和 Moment.js
开发语言·javascript·ecmascript