【发布时间】:2013-07-24 13:05:03
【问题描述】:
假设,我在 Node.js 中有一个异步函数,基本上是这样的:
var addAsync = function (first, second, callback) {
setTimeout(function () {
callback(null, first + second);
}, 1 * 1000);
};
现在我当然可以异步调用这个函数了:
addAsync(23, 42, function (err, result) {
console.log(result); // => 65
});
我想知道的是您是否可以以某种方式同步调用此函数。为此,我想要一个包装函数sync,它基本上做了以下事情:
var sync = function (fn, params) {
var res,
finished = false;
fn.call(null, params[0], params[1], function (err, result) {
res = result;
finished = true;
});
while (!finished) {}
return res;
};
然后,我可以同步运行addAsync,通过这种方式调用它:
var sum = sync(addAsync, [23, 42]);
注意:当然你不会在现实中使用params[0] 和params[1],而是相应地使用arguments 数组,但我想在这个例子中保持简单。
现在,问题是,上面的代码不起作用。它只是阻塞,因为 while 循环阻塞并且不会释放事件循环。
我的问题是:是否有可能以任何方式使该示例按预期运行?
我已经尝试过setImmediate 和process.nextTick 以及其他各种东西,但它们都没有帮助。基本上,您需要一种方法来告诉 Node.js 请暂停当前函数,继续运行事件循环,并在稍后的时间点返回。
我知道你可以使用yield 和生成器函数来实现类似的功能,至少在 Node.js 0.11.2 及更高版本中是这样。但是,我很好奇即使没有它是否也有效?
请注意,我完全了解如何在 Node.js 中进行异步编程、事件循环和所有相关内容。我也完全意识到编写这样的代码是一个坏主意,尤其是在 Node.js 中。而且我也完全意识到“主动等待”也是一个愚蠢的想法。 因此,请不要建议学习如何异步执行此操作或类似的操作。我知道。
我问的原因只是出于好奇和学习的愿望。
【问题讨论】:
-
并非如此。如果你在一段时间内“拿着麦克风”,它就不能执行回调。如果它不能执行回调它就不能设置 res。如果你在 while 上停下来让它可以执行回调,flow 会从外部函数中滑出,并且 res 在定义之前返回。可能有一个 NPM 工具可以编写这样的代码...
-
你是什么意思 可能有一个 NPM 工具来编写类似的代码?
-
伪同步代码,看起来是程序化的,但在内部使用回调来执行您描述的逻辑。我自己用 throw 和 Function 完成了它,但这是一个讨厌的习惯。
-
你有这样的代码示例吗?
标签: javascript node.js asynchronous synchronous event-loop