5

我有以下用 C++ 编写的几乎可以工作的代码:

[..]

Handle<Object> jsGlobal;
Handle<Function> jsUpdateFunc;

void setupJs () {
    V8::Initialize();
    Isolate* isolate = v8::Isolate::New();
    Isolate::Scope isolate_scope(isolate);
    HandleScope handle_scope(isolate);
    Local<Context> context = Context::New(isolate);
    Context::Scope context_scope(context);
    Local<String> source = String::NewFromUtf8(isolate, "var a = 0; function test() { a++; return a.toString(); }");
    Local<Script> script = Script::Compile(source);
    script->Run();

    jsGlobal = context->Global();
    Handle<Value> value = jsGlobal->Get(String::NewFromUtf8(isolate, "test"));
    jsUpdateFunc = Handle<Function>::Cast(value);
}

void callJs() {
    Handle<Value> args[0];
    Handle<Value> js_result = jsUpdateFunc->Call(jsGlobal, 0, args);
    js_result->ToString();

    String::Utf8Value utf8(js_result);
    printf("%s\n", *utf8);
}

[..]

我有函数 setupJs() 设置了 v8 环境,并且 callJs 应该被多次调用(在工作时,javascript 脚本每次将 var a 递增一)。

如果我把

Handle<Value> args[0];
Handle<Value> js_result = jsUpdateFunc->Call(jsGlobal, 0, args);
js_result->ToString();

String::Utf8Value utf8(js_result);
printf("%s\n", *utf8);

在 setupJs 中,我可以看到如何调用函数并打印“1”。但是,如果我将函数调用与稍后调用的不同函数一起保留,我会在该行出现 SegfaultHandle<Value> js_result = jsUpdateFunc->Call(jsGlobal, 0, args);

我已经检查过,jsUpdateFunc 和 jsGlobal 都是非空指针

4

1 回答 1

7

您需要为jsGlobal和使用持久句柄jsUpdateFuncv8::HandleScope普通(本地)句柄在其封闭被破坏时变得无效。

您还需要一个用于v8::Isolate指针的全局变量和另一个用于v8::Context.

要稍后调用脚本函数,您需要:

  1. 锁定隔离区(您也应该这样做setupJs;请参阅v8::Locker
  2. 输入隔离物(参见v8::Isolate::Scope)。
  3. 建立句柄范围(参见 参考资料v8::HandleScope)。
  4. 为上下文创建一个本地句柄。
  5. 输入上下文(参见v8::Context::Scope)。
  6. jsGlobal为和创建本地句柄jsUpdateFunc
  7. 如上所述调用脚本函数。

在 V8 头文件中查找v8::Persistent和相关的模板。

于 2015-10-21T12:28:35.240 回答