我有一个全局事件管理器,允许您使用lambdas监听string
个事件名称。
// somewhere in the ModuleScript class
Event->Listen("WindowResize", [=]{
// ...
});
现在,我也希望从JavaScript注册事件。因此,我写了这个回调。
v8::Handle<v8::Value> ModuleScript::jsOn(const v8::Arguments& args)
{
// get pointer to class since we're in a static method
ModuleScript *module = (ModuleScript*)HelperScript::Unwrap(args.Data());
// get event name we want to register to from arguments
if(args.Length() < 1 || !args[0]->IsString())
return v8::Undefined();
string name = *v8::String::Utf8Value(args[0]);
// get callback function from arguments
if(args.Length() < 2 || !args[1]->IsFunction())
return v8::Undefined();
v8::Handle<v8::Function> callback =
v8::Local<v8::Function>::Cast(args[1]->ToObject());
// register event on global event manager
module->Event->Listen(name, [=]{
// create persistent handle so that function stays valid
// maybe this doesn't work, I don't know
v8::Persistent<v8::Function> function =
v8::Persistent<v8::Function>::New(args.GetIsolate(), callback);
// execute callback function
// causes the access violation
function->Call(function, 0, NULL);
});
return v8::Undefined();
}
触发事件时,应用程序因访问冲突而崩溃。我的想法是,此时函数对象不再有效,或者它是JavaScript范围问题。但我无法理解。
导致访问冲突的原因是什么以及如何克服它?
答案 0 :(得分:2)
我相信这里有几个潜在的问题。
首先,在ModuleScript::jsOn()
终止后,您没有使用持久句柄来保存JavaScript函数。在调用事件处理程序时,该函数可能已经消失。考虑将callback
作为持久句柄。
其次,您的事件处理程序需要在调用JavaScript函数之前输入适当的V8上下文。根据您的体系结构,也可能需要明确锁定和进入V8隔离。
第三(在您的特定方案中这可能不是问题),您需要管理V8隔离的生命周期。如果事件管理器在后台线程上触发事件,则必须确保事件处理程序以某种方式阻止隔离层从另一个线程中释放。不幸的是,这是V8 API无法提供很多帮助的一个领域。
第四,为了防止泄漏,你的事件处理程序应该在调用函数后处理持久性函数句柄。
祝你好运!