如何存储对函数的引用,以便稍后在node.js C++插件模块中调用它?

Dar*_*ite 5 c++ v8 node.js embedded-v8 libuv

这是一个node.js插件模块,我用C++编写,使用node-gyp构建.当StoreFunction我试图存储一个指向函数的指针,以便我以后可以使用它

当我稍后尝试在InvokeFunction中调用它时,我得到一个Segmentation错误.如果我检查两个函数中的指针(使用cout)它们是相同的值,那令我困惑的是什么.

所以我猜测调用两个函数之间调用上下文的变化或者我不明白我指向的是什么.

所有(ummmmmm)指针在这里感谢我的问题..............

#include <node.h>
#include <v8.h>

using namespace v8;
v8::Persistent<v8::Function> callbackFunction;
 Handle<Value> StoreFunction(const Arguments& args) {
    HandleScope scope;
    callbackFunction = *Local<Function>::Cast(args[0]);
    return scope.Close(Undefined());
}

Handle<Value> InvokeFunction(const Arguments& args) {
    HandleScope scope;
    Local<Value> argv[1] = { String::New("Callback from InvokeFunction")};
    callbackFunction->Call(Context::GetCurrent()->Global(), 1, argv);
    return scope.Close(Undefined());
}

void init(Handle<Object> target) {
  NODE_SET_METHOD(target, "StoreFunction", StoreFunction);
  NODE_SET_METHOD(target, "InvokeFunction", InvokeFunction);
}

NODE_MODULE(someaddonmodule, init);
Run Code Online (Sandbox Code Playgroud)

当然还有一些叫js ...........

var myaddon = require('../build/Release/someaddonmodule');
myaddon.StoreFunction(function(data){   
    console.log("Called back: "+data);
});

myaddon.InvokeFunction();   //causes a segmentation fault
Run Code Online (Sandbox Code Playgroud)

Dar*_*ite 3

答案是因为我们不再使用 Java 编程 Toto。我创建的指针指向本地句柄,而不是函数。持有对此的“引用”不足以阻止 V8 垃圾收集在作用域关闭时销毁它。

为了处理这个问题,需要向 V8 发出明确的请求,留出一些内存来保存如下所示的函数:

Persistent< Function > percy;
Local<Function> callbackFunction = Local<Function>::Cast(args[0]);
percy = Persistent<Function>::New(callbackFunction);
Run Code Online (Sandbox Code Playgroud)

如果任何对 V8 内部结构有更好了解的人知道的比这更多,我仍然很想听听你的解释:)