返回

解构Promise.all和Promise.race:从源码到原理

前端

Promise.all

Promise.all方法接受一个Promise对象数组作为参数,并返回一个新的Promise对象。这个新的Promise对象将在所有传入的Promise对象都成功resolve后resolve,或者任何一个传入的Promise对象reject后reject。

源码分析

Promise.all = function(promises) {
  return new Promise((resolve, reject) => {
    const results = [];
    let pending = promises.length;

    for (let i = 0; i < promises.length; i++) {
      promises[i].then((result) => {
        results[i] = result;
        pending--;

        if (pending === 0) {
          resolve(results);
        }
      }).catch((error) => {
        reject(error);
      });
    }
  });
};

从源码中可以看出,Promise.all首先创建一个新的Promise对象,然后遍历传入的Promise对象数组,为每个Promise对象添加一个then回调函数。

then回调函数的作用是将Promise对象的状态和值传递给新的Promise对象。如果Promise对象成功resolve,则将结果存储在results数组中,并减少pending计数。如果Promise对象reject,则直接调用新的Promise对象的reject方法,并将错误信息传递给它。

当所有传入的Promise对象都成功resolve后,pending计数将变为0,此时新的Promise对象将调用resolve方法,并将results数组作为参数传递给它。

Promise.race

Promise.race方法也接受一个Promise对象数组作为参数,但它与Promise.all不同,它返回一个新的Promise对象,该对象将在第一个传入的Promise对象resolve或reject后resolve或reject。

源码分析

Promise.race = function(promises) {
  return new Promise((resolve, reject) => {
    for (let i = 0; i < promises.length; i++) {
      promises[i].then((result) => {
        resolve(result);
      }).catch((error) => {
        reject(error);
      });
    }
  });
};

从源码中可以看出,Promise.race首先创建一个新的Promise对象,然后遍历传入的Promise对象数组,为每个Promise对象添加一个then回调函数。

then回调函数的作用是将Promise对象的状态和值传递给新的Promise对象。如果Promise对象成功resolve,则直接调用新的Promise对象的resolve方法,并将结果作为参数传递给它。如果Promise对象reject,则直接调用新的Promise对象的reject方法,并将错误信息传递给它。

当第一个传入的Promise对象resolve或reject后,新的Promise对象将停止等待其他Promise对象的状态变化,并立即调用resolve或reject方法。

使用场景

Promise.all和Promise.race都有各自的适用场景。

Promise.all通常用于需要等待所有异步任务都完成后再进行后续操作的情况。例如,在需要获取多个资源时,可以使用Promise.all来等待所有资源都获取完毕后,再将这些资源组合成一个新的对象。

Promise.race通常用于需要在多个异步任务中尽快完成一个任务的情况。例如,在需要向多个服务器发送请求时,可以使用Promise.race来等待第一个服务器返回响应,然后立即停止等待其他服务器的响应。

总结

Promise.all和Promise.race都是非常有用的异步编程工具,它们可以帮助我们协调多个异步任务的执行,并根据不同的需求选择合适的工具。