node.js - Nan 无法使用 v8::Object 字段包装和解包对象

标签 node.js v8

一开始我是这样定义的(和文档中的object wrap例子基本一样,唯一的区别是这个例子包装了一个double值的属性,而我的是v8::Object):

.h:

#include <nan.h>

using v8::Local;
using v8::Object;
using Nan::FunctionCallbackInfo;

class MyObject : public Nan::ObjectWrap {
    public:
    static void Init(v8::Local<v8::Object> module);

    private:
    explicit MyObject(Local<Object>);
    ~MyObject();

    static Nan::Persistent<v8::Function> constructor;
    static void New(const FunctionCallbackInfo<v8::Value>& info);
    static void GetConfig(const FunctionCallbackInfo<v8::Value>& info);

    Local<Object> Config;
};

.cc:

using v8::Local;
using v8::Object;

Nan::Persistent<v8::Function> MyObject::constructor;
MyObject::MyObject(Local<Object> config) : Config(config){}
MyObject::~MyObject(){}

// return an object when required
void MyObject::Init(Local<Object> module)
{
    Nan::HandleScope scope;

    Local<v8::FunctionTemplate> tpl = Nan::New<v8::FunctionTemplate>(New);
    tpl->SetClassName(Nan::New("myObject").ToLocalChecked());
    tpl->InstanceTemplate()->SetInternalFieldCount(1);

    // prototypes
    Nan::SetPrototypeMethod(tpl, "getConfig", GetConfig);

    constructor.Reset(tpl->GetFunction());
    module->Set(Nan::New("exports").ToLocalChecked(), tpl->GetFunction());
}

void MyObject::New(const Nan::FunctionCallbackInfo<v8::Value>& info) {
    Local<Object> conf = info[0].As<Object>();
    MyObject* myObject = new MyObject(conf);
    MyObject->Wrap(info.This());
    info.GetReturnValue().Set(info.This());
}

void MyObject::GetConfig(const Nan::FunctionCallbackInfo<v8::Value>& info)
{
    MyObject* myObject = ObjectWrap::Unwrap<MyObject>(info.Holder());
    Local<Object> conf = myObject->Config;

    info.GetReturnValue().Set(conf);
}

js:

var Test = require("./build/Release/test");
var test = new Test({a: 1});
console.log(test.getConfig()); // logs 3276x randomly

似乎 Config 字段已被垃圾收集。如果是这种情况,我不明白为什么,因为 MyObject 实例显然仍在范围内。但我仍然试图让 Config 持久化。

有趣的是,直接将 Config 的类型更改为 Persistent<Object> * 也不起作用,但是当我添加这些看似无关的行以测试我是否在对象被包裹在 MyObject::New() 之前从 js 传递了正确的对象时,它起作用了:

Local<Object> test = Nan::New(*MyObject->Config);
Local<v8::String> v8Str = Nan::To<v8::String> (test->Get(Nan::New("a").ToLocalChecked())).ToLocalChecked();
v8::String::Utf8Value v8StrUtf8(v8Str);
std::string str = std::string(*v8StrUtf8);
std::cout << str << std::endl;

这是怎么回事?包装 v8::Object 的正确方法是什么?为什么那些属性访问线可以让它工作?

最佳答案

如果你想保留一个值更长的时间(超出当前范围),你需要使引用成为持久的而不是本地的。这样做将防止值(value)被垃圾收集。

因此将您的 Config 定义更改为:

Nan::Persistent<Object> Config;

我还没有测试过,但您可能还需要将您的构造函数更改为:

MyObject::MyObject(Local<Object> config) {
  Config.Reset(config);
}

然后,当您想要检索值时,您需要从 Persistent 获取本地句柄,例如:

Local<Object> conf = Nan::New(myObject->Config);

关于node.js - Nan 无法使用 v8::Object 字段包装和解包对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39313262/

相关文章:

javascript - javascript 中数组中值的匹配

javascript - 使用 Google V8 从 C++ 代码中抛出 JavaScript 异常

c++ - 为什么包裹在类里面时 V8 会崩溃

node.js - 如何使用GraphicsMagick for NodeJS制作图片和文字?

javascript - Mongoose 具有相同的保存和更新中间件功能

javascript - 调用堆栈和事件循环 - 为什么要等待空堆栈?

javascript - 为什么通过 Object.defineProperty 添加的属性在 Javascript 中不可迭代

c++ - hello-world.cc 样本似乎泄漏了内存

node.js - 在 AWS Lambda 上运行 SQLite3

node.js - 如何使用 Socket.IO 获取已连接套接字/客户端的列表?