好的,我有一个需要从 JavaScript 调用的 C++ 函数,其中一个参数是 JavaScript 对象。JavaScript 看起来像这样:
var message = {
fieldA: 42,
fieldB: "moo"
};
myObj.send(message, function (err) { console.log("Result: " + err); });
在 send() 例程中,我需要调用另一个可能阻塞的 C 库中的本机函数。这个库中的所有函数都可能阻塞,所以我一直在广泛使用 uv_queue_work。
这个例程是我第一次遇到问题,这是因为 JavaScript 对象。C++ 代码如下所示:
struct SendMessageRequest
{
Persistent<Object> message;
Persistent<Function> callback;
int result;
};
Handle<Value> MyObj::Send(const Arguments& args)
{
HandleScope scope;
// Parameter checking done but not included here
Local<Object> message = Local<Object>::Cast(args[0]);
Local<Function> callback = Local<Function>::Cast(args[1]);
// Send data to worker thread
SendMessageRequest* request = new SendMessageRequest;
request->message = Persistent<Object>::New(message);
request->callback = Persistent<Function>::New(callback);
uv_work_t* req = new uv_work_t();
req->data = request;
uv_queue_work(uv_default_loop(), req, SendMessageWorker, SendMessageWorkerComplete);
return scope.Close(Undefined());
}
这一切都很好,当我尝试在 SendMessageWorker 函数中访问 request->message 时,问题就来了。
void SendMessageWorker(uv_work_t* req)
{
SendMessageRequest* request = (SendMessageRequest*)req->data;
Local<Array> names = request->message->GetPropertyNames();
// CRASH
似乎从 request->message 调用方法会导致在一个非常小的地址上发生访问冲突(可能是 V8/node 中某处的 NULL 指针引用)。所以直接使用request->message肯定是错的。我知道要访问我需要这样做的回调函数:
request->callback->Call(Context::GetCurrent()->Global(), 1, argv);
我是否需要使用 Context::GetCurrent()->Global() 来访问由 Persistent 模板包装的 Object 类的方法?如果是这样,我该怎么做?