一个关于Promise的问题

2021/07/06

一道题带来的疑问

掘金上的一道题: JS 实现一个带并发限制的异步调度器 Scheduler,保证同时运行的任务最多有两个,一个解法如下:

function timeout(time) {
  return new Promise((resolve) => {
    setTimeout(resolve, time);
  });
}
class Scheduler {
  constructor() {
    this.awaitArr = [];
    this.count = 0;
  }

  async add(promiseCreator) {
    if (this.count >= 2) {
      await new Promise((resolve) => {
        this.awaitArr.push(resolve);
      });
    }
    this.count++;
    const res = await promiseCreator();
    this.count--;

    if (this.awaitArr.length) {
      this.awaitArr.shift()();
    }
    return res;
  }
}
const scheduler = new Scheduler();
function addTask(time, order) {
  scheduler.add(() => timeout(time)).then(() => console.log(order));
}
addTask(1000, '1000');
addTask(500, '500');
addTask(400, '400');
addTask(300, '300');
// 输出
// 500
// 400
// 1000
// 300

初看有些懵,this.awaitArr.push(resolve);,好奇为什么不是吧promiseCreatorpush 进去,仔细看下这个 Promise 里面的函数没有执行 resolve 函数,只是把 resolve 函数 push 进数组了。那么问题来了!

如果 Promise 的 resolve, reject 没有执行会怎么样?

执行以下代码

let tAsync = async function () {
    console.log(`----->`)
    await new Promise(() => {});
    console.log(`<-----`)
};

tAsync();
//输出
----->

发现程序执行结束后并没有输出<-----,

根据 Promise 定义MDN:

new Promise( function(resolve, reject) {...} /* executor */  );

executor 是带有 resolve 和 reject 两个参数的函数。 Promise 构造函数执行时立即调用 executor 函数,resolve 和 reject 两个函数作为参数传递给 executor(executor 函数在 Promise 构造函数返回所建 promise 实例对象前被调用)。resolve 和 reject 函数被调用时,分别将 promise 的状态改为 fulfilled(完成)或 rejected(失败)。如果在 executor 函数中抛出一个错误,那么该 promise 状态为 rejected。executor 函数的返回值被忽略。

看来由于 await 使异步功能的执行被暂停,直到 Promise 被解决(resolve 或 reject),并在实现后恢复执行异步功能。由于该 Promise 一直没有调用 resolve 和 reject 导致下面的程序无法执行。看到这又有个疑问?

在 Promise 的外部执行 resolve, reject 可以改变 Promise 的状态吗?

我们看到解法中的 resolve 在 Promise 外部被执行,这样是否可以改变 Promise 的状态呢? 将上面代码改造一下,验证看看:

let waitf = null;
const tAsync = async function () {
  console.log(`----->`);
  await new Promise((resolve) => {
    waitf = resolve;
  });
  console.log(`<-----`);
};
tAsync();
waitf();

// 输出
// ----->
// <-----

结论

这时再回头看看上面的代码就容易理解了