Skip to content

Generator

概念

在ES6(ECMAScript 2015)中,JavaScript引入了一种新的函数类型:Generator函数。

Generator函数是可以暂停执行并在稍后恢复的特殊函数,这种行为由yield关键字控制。

当函数遇到yield语句时,它将暂停执行,并将紧跟在yield后的值作为返回结果。下面是一个简单的示例:

javascript
function* generatorFunction() {
  yield 'Hello, ';
  yield 'World!';
}

const generator = generatorFunction();

console.log(generator.next().value); // 'Hello, '
console.log(generator.next().value); // 'World!'
console.log(generator.next().done); // true

在这个例子中,generatorFunction是一个Generator函数。调用这个函数不会直接执行函数体内的代码,而是返回一个Generator对象。调用Generator对象的next方法,函数体内的代码将从头开始执行,或者从上一次yield语句处继续执行,直到遇到下一个yield语句。每次调用next方法,都会返回一个对象,包含valuedone两个属性。value属性是yield语句后面的值,done属性表示函数是否执行完成。

这种暂停执行的特性使得Generator函数能够以一种完全不同的方式来编写和理解代码,尤其是在处理复杂的异步逻辑时。

异步操作

Generator函数的真正威力在于它能以同步的方式来编写异步代码。通过使用yield关键字,我们可以暂停函数的执行,等待异步操作完成,然后再继续执行。

这是一个使用Generator函数处理异步操作的例子:

javascript
function* fetchUser(userId) {
  const response = yield fetch(`https://api.example.com/users/${userId}`);
  const user = yield response.json();
  return user;
}

const generator = fetchUser(1);
const responsePromise = generator.next().value;

responsePromise.then(response => {
  const userPromise = generator.next(response).value;
  userPromise.then(user => generator.next(user));
});

在这个例子中,我们首先发起一个网络请求来获取用户信息。这是一个异步操作,但是使用yield关键字,我们可以将其转化为一个同步操作。网络请求完成后,我们获取响应并解析为JSON。这也是一个异步操作,但是我们同样可以使用yield关键字来将其转化为同步操作。

实现Async/Await

在JavaScript中,Async/Await是一种处理异步操作的新方法,它基于Promise和Generator函数。实际上,我们可以使用Generator函数来模拟Async/Await的行为。

首先,我们需要一个处理Generator函数的辅助函数,来自动执行Generator函数:

javascript
function asyncGenerator(generatorFunc) {
  return function (...args) {
    const generator = generatorFunc(...args);

    function handle(result) {
      if (result.done) return Promise.resolve(result.value);
      return Promise.resolve(result.value)
        .then(res => handle(generator.next(res)))
        .catch(err => handle(generator.throw(err)));
    }

    return handle(generator.next());
  };
}

这个asyncGenerator函数接受一个Generator函数作为参数,返回一个新的函数。当这个新的函数被调用时,它首先创建一个Generator对象。然后,它定义了一个handle函数来处理Generator对象的返回结果。如果Generator函数已经执行完毕,它将返回一个解析为最后返回值的Promise;如果Generator函数还未执行完毕,它将处理当前的Promise,等待Promise解析完成后再次调用handle函数。这样,我们就可以像使用Async/Await那样使用Generator函数。

接下来,我们可以使用asyncGenerator函数来改写前面的fetchUser函数:

javascript
const fetchUser = asyncGenerator(function* (userId) {
  const response = yield fetch(`https://api.example.com/users/${userId}`);
  const user = yield response.json();
  return user;
});

fetchUser(1).then(user => console.log(user));

这段代码的行为与使用Async/Await完全相同。实际上,Async/Await在底层就是使用了类似的机制。