我正在使用v8为我的C ++应用程序创建一个javascript界面,但遇到了函数回调的问题。
我有一个对象模板,它有一个setter和getter用于对象“update”,它只是设置/获取setter和getter都可以访问的对象句柄(参见“我尝试过的东西”。)一个对象被实例化在此对象模板中称为“world”的全局上下文中。然后运行一个脚本,将“world.update”设置为具有基本输出消息的函数。程序然后获取更新功能并调用它完全按预期工作 - 打印一些输出。程序然后再次获取更新功能,但更新功能现在是一个字符串 - 原始调用的输出。试图调用它会导致异常。
代码:
#include <iostream>
#include <fstream>
#include <string>
#include <v8.h>
using namespace v8;
std::string readFile(std::string fname) {
std::ifstream finput(fname);
std::string filestr((std::istreambuf_iterator<char>(finput)),
std::istreambuf_iterator<char>());
return filestr;
}
Handle<Value> print(const Arguments& args) {
String::AsciiValue str(args[0]);
std::cout << *str;
HandleScope scope;
return scope.Close(Undefined());
}
class FuncWrapper {
public:
Handle<Function> func;
};
Handle<Value> getWorldUpdate(Local<String> property, const AccessorInfo &info) {
std::cout << "Get update() [" << *String::AsciiValue(property) << "]\n";
Local<Object> self = info.Holder();
Local<External> wrap = Local<External>::Cast(self->GetInternalField(0));
FuncWrapper *fw = static_cast<FuncWrapper*>(wrap->Value());
return fw->func;
}
void setWorldUpdate(Local<String> property, Local<Value> value, const AccessorInfo& info) {
std::cout << "Set update() [" << *String::AsciiValue(property) << "]\n";
Local<Object> self = info.Holder();
Local<External> wrap = Local<External>::Cast(self->GetInternalField(0));
FuncWrapper *fw = static_cast<FuncWrapper*>(wrap->Value());
//Accessor info could be used to get the class here
fw->func = Handle<Function>::Cast(value);
}
int main() {
// Create a stack-allocated handle scope.
HandleScope handle_scope;
//Add stuff
Handle<ObjectTemplate> globalScope = ObjectTemplate::New();
globalScope->Set(String::New("print"), FunctionTemplate::New(print));
Handle<ObjectTemplate> worldTmpl = ObjectTemplate::New();
worldTmpl->SetInternalFieldCount(1);
worldTmpl->SetAccessor(String::New("update"), getWorldUpdate, setWorldUpdate);
// Create a new context.
Handle<Context> context = Context::New(NULL, globalScope);
// Enter the created context for compiling
Context::Scope context_scope(context);
Handle<Object> global = context->Global();
Handle<Object> world = worldTmpl->NewInstance();
FuncWrapper worldUpdateFunc;
world->SetInternalField(0, External::New((void*)&worldUpdateFunc));
global->Set(String::New("world"), world);
// Compile the source code.
Handle<Script> script = Script::Compile(String::New(readFile("main.js").c_str()));
// Run the script to get the result.
script->Run();
v8::TryCatch try_catch;
Handle<Function> updateFunc = Handle<Function>::Cast(world->Get(String::New("update")));
updateFunc->Call(updateFunc, 0, NULL);
if (try_catch.HasCaught()) {
String::AsciiValue asciistr(try_catch.Message()->Get());
std::cout << "Caught1: " << *asciistr << "\n";
return -1;
}
//Re-calling. Has the same effect as calling worldUpdateFunc.func
updateFunc = Handle<Function>::Cast(world->Get(String::New("update")));
updateFunc->Call(updateFunc, 0, NULL);
if (try_catch.HasCaught()) {
String::AsciiValue asciistr(try_catch.Message()->Get());
std::cout << "Caught2: " << *asciistr << "\n";
return -1;
}
return 0;
}
脚本(main.js):
"use strict";
world.update = function() {
print("Did a world.update()\n");
}
输出:
Set update() [update]
Get update() [update]
Did a world.update()
Get update() [update]
Caught2: Uncaught TypeError: Did a world.update()
is not a function
没有对象模板(即javascript中没有getter / setter组合的常规对象),程序运行正常,但我希望能够使用它来让脚本管理回调。
为什么会发生这种情况,我做错了什么?
我尝试过的事情:
答案 0 :(得分:0)
问题是正在删除函数句柄指向的对象。
解决方案只是包装一个Persistent而不是一个句柄,然后调用Persistent :: New()来创建一个新的函数实例。
class FuncWrapper {
public:
Persistent<Function> func;
};
...
void setWorldUpdate(Local<String> property, Local<Value> value, const AccessorInfo& info) {
...
fw->func = Persistent<Function>::New(Local<Function>::Cast(value));
}
感谢v8-users group的人帮助我solve this!