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
});
如果您现在承诺的值是一个数组,则可以执行 reduce
或 map
实现。但在这种情况下,您不需要发电机。
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();
}
forEach
实施:
function forEach(promise, callback) {
return promise.then(function(iterable) {
for (var x of iterable) {
callback(x);
}
});
}
所以如果你有这样的函数:
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
});
如果您现在承诺的值是一个数组,则可以执行 reduce
或 map
实现。但在这种情况下,您不需要发电机。
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();
}
forEach
实施:
function forEach(promise, callback) {
return promise.then(function(iterable) {
for (var x of iterable) {
callback(x);
}
});
}