我正在尝试将V8嵌入多线程应用程序中。我的计划是在当前需要执行JS代码的任何线程中以内联方式执行V8代码。
因此,在执行上下文中,我要获取一个v8::Locker
并在 random 线程中输入隔离。我注意到,获取v8::Locker
时,隔离丢失是CurrentContext。
在下面的代码片段中,我将其归结为问题的实质。如果我在v8 :: Locker下Enter
使用上下文,退出该储物柜并输入一个新的储物柜,则隔离将丢失其Context。请注意,我从不明确Exit
上下文。
这是正确用法吗?我想念什么吗?
在下面的代码片段中,您可以看到问题的单线程虚拟示例。 断言 1 和 2 通过并断言 3 失败。
v8::V8::InitializeICU(V8_ICU_FILE);
v8::V8::InitializeExternalStartupData(V8_SNAPSHOT_BIN_FILE);
std::unique_ptr<v8::Platform> platform = v8::platform::NewDefaultPlatform();
v8::V8::InitializePlatform(platform.get());
v8::V8::Initialize();
// Create a new Isolate and make it the current one.
v8::Isolate::CreateParams create_params;
auto allocator = std::unique_ptr<v8::ArrayBuffer::Allocator>(v8::ArrayBuffer::Allocator::NewDefaultAllocator());
create_params.array_buffer_allocator = allocator.get();
v8::Isolate* isolate = v8::Isolate::New(create_params);
v8::HandleScope handle_scope(isolate);
// Create a new context.
v8::Local<v8::ObjectTemplate> global = v8::ObjectTemplate::New(isolate);
global->Set(isolate, "log", v8::FunctionTemplate::New(isolate, logCallback));
v8::Local<v8::Context> context = v8::Context::New(isolate, nullptr, global);
// Enter the context for compiling and running the hello world script.
{
v8::Locker locker(isolate);
v8::Isolate::Scope scope(isolate);
context->Enter(); // Explicitly Enter the context (Which makes it the CurrentContext)
assert(!isolate->GetCurrentContext().IsEmpty()); // 1 - Succeeds
isolate->Exit();
isolate->Enter();
assert(!isolate->GetCurrentContext().IsEmpty()); // 2 - Succeeds
}
{
v8::Locker locker(isolate);
v8::Isolate::Scope scope(isolate);
assert(!isolate->GetCurrentContext().IsEmpty()); // 3 - Assertion Fails
}
// Create a string containing the JavaScript source code.
isolate->Dispose();
v8::V8::Dispose();
v8::V8::ShutdownPlatform();
答案 0 :(得分:0)
v8::Locker
析构函数假定当前线程已完成所有工作,然后进行清理;特别是这意味着它将取消当前上下文。在下一个线程上创建下一个Locker之后,您必须(重新)输入正确的上下文。 v8::Context::Scope
方便了。