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

作者 : 开心源码 本文共4428个字,预计阅读时间需要12分钟 发布时间: 2022-05-13 共182人阅读

原文发布在:手写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」 的作者。您可以通过Watch or Star文章仓库 「github.com/dongyuanxin/blog」,或者关注公众号「心谭博客」来接收最新文章消息。

过程

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;      })  );};

参考链接

  • 文中的代码和用例均在:github.com/dongyuanxin/diy-promise
  • 《ECMAScript 6 入门-Promise 对象》
  • github.com/tc39/proposal-promise-allSettled
  • github.com/matthew-andrews/Promise.prototype.finally

假如觉得有收获,欢迎Watch or Star文章仓库 「github.com/dongyuanxin/blog」,或者扫码关注公众号「心谭博客」,解锁更多文章。

image

说明
1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长!
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是摆设,本站源码仅提供给会员学习使用!
7. 如遇到加密压缩包,请使用360解压,如遇到无法解压的请联系管理员
开心源码网 » 「超全」手写Promise的相关方法

发表回复