「超全」手写Promise的相关方法

原文发布在:手写Promise的相关方法

摘要

Promise 作为 JS 社区的异步解决方案,为开发者提供了.then()Promise.resolve()Promise.reject()等基本方法。除此之外,为了更方便地组合和控制多个的 Promise 实例,也提供了.all().race()等方法。

本文会在 Promise 的基本方法上,手动实现更高级的方法,来加深对 Promise 的理解:

  • 🤔️ 实现Promise.all
  • 🤔️ 实现Promise.race
  • 🤔️ 实现Promise.any
  • 🤔️ 实现Promise.allSettled
  • 🤔️ 实现Promise.finally

⚠️ 完整代码和用例请到github.com/dongyuanxin/diy-promise

实现 Promise.all

过程

Promise.all(iterators)返回一个新的 Promise 实例。iterators 中包含外界传入的多个 promise 实例。

对于返回的新的 Promise 实例,有以下两种情况:

  • 如果传入的所有 promise 实例的状态均变为fulfilled,那么返回的 promise 实例的状态就是fulfilled,并且其 value 是 传入的所有 promise 的 value 组成的数组。
  • 如果有一个 promise 实例状态变为了rejected,那么返回的 promise 实例的状态立即变为rejected

代码实现

实现思路:

  • 传入的参数不一定是数组对象,可以是"遍历器"
  • 传入的每个实例不一定是 promise,需要用Promise.resolve()包装
  • 借助"计数器",标记是否所有的实例状态均变为fulfilled
Promise.myAll = function(iterators) {
  const promises = Array.from(iterators);
  const num = promises.length;
  const resolvedList = new Array(num);
  let resolvedNum = 0;

  return new Promise((resolve, reject) => {
    promises.forEach((promise, index) => {
      Promise.resolve(promise)
        .then(value => {
          // 保存这个promise实例的value
          resolvedList[index] = value;
          // 通过计数器,标记是否所有实例均 fulfilled
          if (++resolvedNum === num) {
            resolve(resolvedList);
          }
        })
        .catch(reject);
    });
  });
};

实现 Promise.race

过程

Promise.race(iterators)的传参和返回值与Promise.all相同。但其返回的 promise 的实例的状态和 value,完全取决于:传入的所有 promise 实例中,最先改变状态那个(不论是fulfilled还是rejected)。

代码实现

实现思路:

  • 某传入实例pending -> fulfilled时,其 value 就是Promise.race返回的 promise 实例的 value
  • 某传入实例pending -> rejected时,其 error 就是Promise.race返回的 promise 实例的 error
Promise.myRace = function(iterators) {
  const promises = Array.from(iterators);

  return new Promise((resolve, reject) => {
    promises.forEach((promise, index) => {
      Promise.resolve(promise)
        .then(resolve)
        .catch(reject);
    });
  });
};

实现 Promise.any

关注微信公众号 「心谭博客」,前往 「xin-tan.com
专注前端与算法的系列干货分享,欢迎关注(¬‿¬)

过程

Promise.any(iterators)的传参和返回值与Promise.all相同。

如果传入的实例中,有任一实例变为fulfilled,那么它返回的 promise 实例状态立即变为fulfilled;如果所有实例均变为rejected,那么它返回的 promise 实例状态为rejected

⚠️Promise.allPromise.any的关系,类似于,Array.prototype.everyArray.prototype.some的关系。

代码实现

实现思路和Promise.all及其类似。不过由于对异步过程的处理逻辑不同,因此这里的计数器用来标识是否所有的实例均 rejected

Promise.any = function(iterators) {
  const promises = Array.from(iterators);
  const num = promises.length;
  const rejectedList = new Array(num);
  let rejectedNum = 0;

  return new Promise((resolve, reject) => {
    promises.forEach((promise, index) => {
      Promise.resolve(promise)
        .then(value => resolve(value))
        .catch(error => {
          rejectedList[index] = error;
          if (++rejectedNum === num) {
            reject(rejectedList);
          }
        });
    });
  });
};

实现 Promise.allSettled

过程

Promise.allSettled(iterators)的传参和返回值与Promise.all相同。

根据ES2020,此返回的 promise 实例的状态只能是fulfilled。对于传入的所有 promise 实例,会等待每个 promise 实例结束,并且返回规定的数据格式。

如果传入 a、b 两个 promise 实例:a 变为 rejected,错误是 error1;b 变为 fulfilled,value 是 1。那么Promise.allSettled返回的 promise 实例的 value 就是:

[{ status: "rejected", value: error1 }, { status: "fulfilled", value: 1 }];

代码实现

实现中的计数器,用于统计所有传入的 promise 实例。

const formatSettledResult = (success, value) =>
  success
    ? { status: "fulfilled", value }
    : { status: "rejected", reason: value };

Promise.allSettled = function(iterators) {
  const promises = Array.from(iterators);
  const num = promises.length;
  const settledList = new Array(num);
  let settledNum = 0;

  return new Promise(resolve => {
    promises.forEach((promise, index) => {
      Promise.resolve(promise)
        .then(value => {
          settledList[index] = formatSettledResult(true, value);
          if (++settledNum === num) {
            resolve(settledList);
          }
        })
        .catch(error => {
          settledList[index] = formatSettledResult(false, error);
          if (++settledNum === num) {
            resolve(settledList);
          }
        });
    });
  });
};

Promise.all、Promise.any 和 Promise.allSettled 中计数器使用对比

这三个方法均使用了计数器来进行异步流程控制,下面表格横向对比不同方法中计数器的用途,来加强理解:

方法名用途
Promise.all标记 fulfilled 的实例个数
Promise.any标记 rejected 的实例个数
Promise.allSettled标记所有实例(fulfilled 和 rejected)的个数

实现 Promise.prototype.finally

过程

它就是一个语法糖,在当前 promise 实例执行完 then 或者 catch 后,均会触发。

举个例子,一个 promise 在 then 和 catch 中均要打印时间戳:

new Promise(resolve => {
  setTimeout(() => resolve(1), 1000);
})
  .then(value => console.log(Date.now()))
  .catch(error => console.log(Date.now()));

现在这段一定执行的共同逻辑,就可以用finally简写为:

new Promise(resolve => {
  setTimeout(() => resolve(1), 1000);
}).finally(() => console.log(Date.now()));

可以看出,Promise.prototype.finally 的执行与 promise 实例的状态无关,不依赖于 promise 的执行后返回的结果值。其传入的参数是函数对象。

代码实现

实现思路:

  • 考虑到 promise 的 resolver 可能是个异步函数,因此 finally 实现中,要通过调用实例上的 then 方法,添加 callback 逻辑
  • 成功透传 value,失败透传 error
Promise.prototype.finally = function(cb) {
  return this.then(
    value => Promise.resolve(cb()).then(() => value),
    error =>
      Promise.resolve(cb()).then(() => {
        throw error;
      })
  );
};

参考链接

专注前端与算法的系列干货分享,欢迎关注(¬‿¬)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Promise方法是指通过自己编代码实现Promise类的相关方法,比如Promise.all和Promise.any。在引用和引用中,作者给出了Promise.all和Promise.any的具体实现代码。 对于Promise.all方法实现,可以使用一个累加器count来判断是否所有的promise都已经成功,并将成功的结果存放在结果数组result中。当累加器count等于传入的promise数组长度时,表示所有的promise都已经成功,可以通过resolve来返回最终的成功结果。如果其中有一个promise失败,则直接通过reject返回失败结果。这里使用了Promise.resolve方法将传入的参数转化为promise对象,以便统一处理。具体的实现代码可以参考引用中的myAll方法。 对于Promise.any方法实现,同样可以使用一个累加器count来判断是否已经找到了成功的promise,并将成功的结果通过resolve返回。如果遍历完所有的promise都没有找到成功的,则通过reject返回失败结果。具体的实现代码可以参考引用中的myAny方法。 需要注意的是,上述的实现代码可能只是简单的示例,并不一定适用于所有情况,实际的实现可能还需要考虑更多的细节和边界条件。另外,Promise方法主要是为了更好地理解Promise的工作原理和实现方式,实际开发中通常会直接使用JavaScript中内置的Promise对象。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [promise方法(all、race、allSettled、any、finally),你会几个](https://blog.csdn.net/weixin_45774485/article/details/122462081)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值