在v8和Node.js中保存函数回调

时间:2012-10-15 01:04:44

标签: javascript node.js v8

如何在v8包装对象中保存javascript提供的回调,以便将来不仅在当前函数调用中使用。基本上我想用C ++创建一个javascript对象,当用新的Object()创建时,提供一个函数回调。然后在整个c ++对象生命中使用该回调。见下面的例子:

我遇到的问题是,当我尝试在不同的静态函数中使用Handle对象时,它会出错。

在节点js文件中:

var Object = require("./customModule");
var obj = new Object(function(){console.log("Callback called...")})

// Emit callback
obj.emitCallback();

在c ++模块标题中

class Object : public node::ObjectWrap {

public:

    static void Init(v8::Handle<v8::Object> target);

    Object();

protected:

    v8::Handle<v8::Function> m_faceDetectCallback;

    static v8::Handle<v8::Value> New(const v8::Arguments& args);

    static v8::Handle<v8::Value> onEmitCallback(const v8::Arguments& args);
}



v8::Handle<v8::Value> Object::New(const v8::Arguments& args) {

    HandleScope scope;

    Object* obj = new Object();
    obj->Wrap(args.This());

    obj->m_faceDetectCallback = Handle<Function>::Cast(args[0]);

    //obj->m_faceDetectCallback = v8::Persistent<Function>::Cast(args[0]);

    // Works fine here.
    const unsigned argc = 1;
    Local<Value> argv[argc] = { Local<Value>::New(String::New("hello world")) };
    obj->m_faceDetectCallback->Call(Context::GetCurrent()->Global(), argc, argv);

    return args.This();
}

static v8::Handle<v8::Value> Object::onEmitCallback(const v8::Arguments& args){
    HandleScope scope;

    Object* obj = ObjectWrap::Unwrap<Object>(args.This());

    const unsigned argc = 1;
    Local<Value> argv[argc] = { Local<Value>::New(String::New("hello world")) };

        //!! Segfaults here
    if(obj->m_faceDetectCallback->IsCallable()){
        //obj->m_faceDetectCallback->Call(Context::GetCurrent()->Global(), argc, argv);
    }


    return scope.Close(v8::String::New("Start called"));
}

1 个答案:

答案 0 :(得分:13)

您需要使用v8::Persistent而不是标准句柄。 HandleLocalPersistent的基类,所以通过执行你正在进行的转换,你抓住了一个指向v8::Function但没有做任何事情的指针V8不要垃圾收集它。

在课堂上有这个:

v8::Persistent<v8::Function> m_faceDetectCallback;

并指定

obj->m_faceDetectCallback = v8::Persistent<v8::Function>::New(args[0]);