from async to generator

想要实现 async 函数的功能,首先要明白 async 函数有哪些特点。

  • 返回 Promise
  • async 函数中可以使用 await 等待 Promise

以上,我们可以使用 generator 来模拟实现 async 函数,因为 yield 同样可以用来暂停恢复函数的执行。

下面我们来看看 babel 是怎么转换 async/await 的。

const xxx = async () => {
  const val = await yyy();
  return val;
}

先分析下原始代码,定义一个async 函数 xxx,这个 async 函数里面使用 await 取得 async 函数 yyy 的值赋值给 val 并返回 val。

以下是经 babel 转换得到的代码:

function _asyncToGenerator(fn) {
  return function() {
    var gen = fn.apply(this, arguments);
    return new Promise(function(resolve, reject) {
      function step(key, arg) {
        try {
          var info = gen[key](arg);
          var value = info.value;
        } catch (error) {
          reject(error);
          return;
        }
        if (info.done) {
          resolve(value);
        } else {
          return Promise.resolve(value).then(
            function(value) {
              step("next", value);
            },
            function(err) {
              step("throw", err);
            }
          );
        }
      }
      return step("next");
    });
  };
}

const xxx = (() => {
  var _ref = _asyncToGenerator(function*() {
    const val = yield yyy();
    return val;
  });

  return function xxx() {
    return _ref.apply(this, arguments);
  };
})();

可以看到 xxx 使用立即执行函数的返回值初始化。在立即执行函数中,原始代码中的 async 函数的内容被放在 generator 中,并被 _asyncToGenerator 处理,await 变成了 yield。

现在来看函数 _asyncToGenerator。我们知道 async 函数会返回一个 Promise,所以 _asyncToGenerator 也要返回一个 Promise。在这个 Promise 中,会重复的执行 gen.next() 直到 done 值为 true,这时 resolve(value) 即可。