ES6 生成器能否生成正常传递给 callback/promise 的数据?

Can ES6 generators yield the data passed normally to a callback/promise?

所以如果你有这样的函数:

function* queryDB(query){
    yield db.collection.find(query);
}

据我了解,它将产生 db.collection.find() returns,所以说它 returns 一个承诺,然后 queryDB() 应该 return 一个可迭代的一个承诺作为其中的一个价值,所以像 co 这样的东西可以用来将承诺合并成一个单一的承诺。有没有一种方法可以在不修改 db.collection.find() 的情况下产生传递给该承诺的内容,以便我可以只使用 for-of 循​​环迭代数据库中的结果?

待澄清

我在问是否可以将传递给函数的参数传递到承诺的 .then() 中。如

queryDB().then(function(<values>)

但我生成了 <values>.

的迭代器

只能在生成器中做到

function* queryDB(query) {
  // Here is synchrous-like code!
  var rows = yield db.collection.find(query);

  var sum = 0;

  // Use the rows synchrously
  for (var x of rows) {
    sum += x;
  }

  return sum;
}

var sumPromise = doM(queryDB('query example'));

sumPromise.then(console.log.bind(console));

参见 demo(我建议尝试一下,看看它在不同情况下的 return 是什么,它会很清楚)。


而你做不到:

for (var x of db.collection.find(query))
  ...

或者:

for (var x of queryDB('some query'))
  ...

它将像:

for (var x of (new Promise(...)))
  ...

您的生成器/包装器将 return 一个承诺。你只能用它做 then


而你做不到:

function* queryDB(query) {
  db.collection.find(query).then(function(values) {
      yield values;
  });
}

最后的解释:

编译器将如何看待这段代码:

function* queryDB(query) {
  db.collection.find(query).then(anonymousFunction);
}

anonymousFunction不是生成器,所以你不能在那里写yield。此外,then 中的 Promise 不需要生成器。

另一方面,如果您想使用 for (var x in queryDB()),编译器期望 return 来自 queryDB() 的数组。如果您的 queryDB() 是异步的,它将 return 一个 Promise-like 接口。

如果换成更函数式风格,可以这样写:

 forEach(queryDB(), function(item) {
    // action on item
 });

如果您现在承诺的值是一个数组,则可以执行 reducemap 实现。但在这种情况下,您不需要发电机。


doM 实施:

function doM(gen) {
  function step(value) {
    var result = gen.next(value);
    if (result.done) {
      return result.value;
    }
    return result.value.then(step);
  }
  return step();
}

来源:Monads in JavaScript.

forEach 实施:

function forEach(promise, callback) {
  return promise.then(function(iterable) {
    for (var x of iterable) {
      callback(x);
    }
  });
}