node.js: 如何从 C++ 模块中产生?

node.js: How to yield from within a C++ module?

背景:v8 支持 yield(旧闻,我知道),这很好地消除了 javascript 代码中使用的回调node.js(参见 https://wingolog.org/archives/2013/05/08/generators-in-v8

问题:由于javascript协程可以调用C++代码(通过模块),被调用的C++如何执行yield操作?举例说明:

// javascript
function* values()
{
    yield 27;
    mycppmodule.someFunction();
}

// c++
mycppmodule::someFunction()
{
    __somehow_yield( 28 ); // how can we make this happen?
}

// user of the code above
var o = values();
o.next(); //  returns 27 - came from javascript above
o.next(); //  returns 28  - came from c++ above, which is invoked by js

我怀疑答案在 V8 API (https://v8docs.nodesource.com/node-7.4/) 的某处,但我的搜索没有产生(双关语)任何结果...

您的 mycppmodule.someFunction(); 无法从生成器函数 values() 返回的生成器中生成值,即使它是在 JavaScript 中编写的。如果你想在 C++ 中做一些像 JavaScript 中的其他代码一样工作的东西,那么你必须首先确保它能在 JavaScript 中工作。

如果你想使用基于生成器的协同程序(比如 co 模块或 Bluebirds 的 coroutine),那么情况有点不同——任何协同程序只是 returns 一个承诺你从生成器中产生的实际上是一个你想要解析并注入生成器的下一个 运行 的承诺,但这里似乎不是这种情况。

话虽如此,首先要确保你的想法可以在 JsvaScript 中实现,我认为:

function* values() {
    yield 27;
    someFunction();
}

let someFunction = // fill the blanks

var o = values();
o.next(); // returns 27 - came from generator
o.next(); // returns 28 - came from someFunction()

如果不将 values() 生成器函数更改为类似以下内容,则无法实现:

function* values() {
    yield 27;
    let gen = someFunction();
    yield gen.next();
}

或:

function* values() {
    yield 27;
    yield* someFunction();
}

如果您可以更改原始 values() 函数,请继续阅读。

现在,生成器函数所做的只是returns一个作为生成器的对象。该生成器具有 .next().throw().return() 等方法。如果您在 C++ 中创建一个具有正确接口的对象,那么您可以将它用作 JavaScript 中的生成器,并使用 yield* 之类的关键字,但我必须对其进行测试以确保它。

查看文档: