Node.js v21.7.2 文档
- Node.js v21.7.2
- ► 目录
-
► 索引
- 断言测试
- 异步上下文跟踪
- 异步钩子
- 缓冲区
- C++ 插件
- 使用 Node-API 的 C/C++ 插件
- C++ 嵌入器 API
- 子进程
- 集群
- 命令行选项
- 控制台
- Corepack
- 加密
- 调试器
- 已弃用的 API
- 诊断通道
- DNS
- 域
- 错误
- 事件
- 文件系统
- 全局变量
- HTTP
- HTTP/2
- HTTPS
- 检查器
- 国际化
- 模块:CommonJS 模块
- 模块:ECMAScript 模块
- 模块:
node:module
API - 模块:包
- 网络
- 操作系统
- 路径
- 性能钩子
- 权限
- 进程
- Punycode
- 查询字符串
- 读取行
- REPL
- 报告
- 单一可执行应用程序
- 流
- 字符串解码器
- 测试运行器
- 计时器
- TLS/SSL
- 跟踪事件
- TTY
- UDP/数据报
- URL
- 实用程序
- V8
- VM
- WASI
- Web Crypto API
- Web Streams API
- 工作线程
- Zlib
- ► 其他版本
- ► 选项
C++ 嵌入器 API#
Node.js 提供了一些 C++ API,可用于从其他 C++ 软件在 Node.js 环境中执行 JavaScript。
这些 API 的文档可以在 Node.js 源代码树中的 src/node.h 中找到。除了 Node.js 公开的 API 之外,V8 嵌入器 API 还提供了一些必要的概念。
由于将 Node.js 作为嵌入式库使用与编写由 Node.js 执行的代码不同,因此重大更改不遵循典型的 Node.js 弃用策略,并且可能在每个 semver-major 版本发布时发生,恕不另行通知。
示例嵌入应用程序#
以下部分将概述如何使用这些 API 从头开始创建一个应用程序,该应用程序将执行等效于 node -e <code>
的操作,即获取一段 JavaScript 代码并在 Node.js 特定环境中运行它。
完整的代码可以在 Node.js 源代码树中找到。
设置每个进程状态#
Node.js 需要一些每个进程状态管理才能运行
- 用于 Node.js CLI 选项 的参数解析,
- V8 每个进程的要求,例如
v8::Platform
实例。
以下示例展示了如何设置这些内容。一些类名分别来自 node
和 v8
C++ 命名空间。
int main(int argc, char** argv) {
argv = uv_setup_args(argc, argv);
std::vector<std::string> args(argv, argv + argc);
// Parse Node.js CLI options, and print any errors that have occurred while
// trying to parse them.
std::unique_ptr<node::InitializationResult> result =
node::InitializeOncePerProcess(args, {
node::ProcessInitializationFlags::kNoInitializeV8,
node::ProcessInitializationFlags::kNoInitializeNodeV8Platform
});
for (const std::string& error : result->errors())
fprintf(stderr, "%s: %s\n", args[0].c_str(), error.c_str());
if (result->early_return() != 0) {
return result->exit_code();
}
// Create a v8::Platform instance. `MultiIsolatePlatform::Create()` is a way
// to create a v8::Platform instance that Node.js can use when creating
// Worker threads. When no `MultiIsolatePlatform` instance is present,
// Worker threads are disabled.
std::unique_ptr<MultiIsolatePlatform> platform =
MultiIsolatePlatform::Create(4);
V8::InitializePlatform(platform.get());
V8::Initialize();
// See below for the contents of this function.
int ret = RunNodeInstance(
platform.get(), result->args(), result->exec_args());
V8::Dispose();
V8::DisposePlatform();
node::TearDownOncePerProcess();
return ret;
}
每个实例状态#
Node.js 有一个“Node.js 实例”的概念,通常被称为 node::Environment
。每个 node::Environment
都与以下内容相关联:
- 正好一个
v8::Isolate
,即一个 JS 引擎实例, - 正好一个
uv_loop_t
,即一个事件循环,以及 - 多个
v8::Context
,但只有一个主v8::Context
。 - 一个
node::IsolateData
实例,其中包含可能由使用相同v8::Isolate
的多个node::Environment
共享的信息。目前,没有针对这种情况进行测试。
为了设置 v8::Isolate
,需要提供一个 v8::ArrayBuffer::Allocator
。一种可能的选择是默认的 Node.js 分配器,可以通过 node::ArrayBufferAllocator::Create()
创建。使用 Node.js 分配器可以在附加组件使用 Node.js C++ Buffer
API 时进行一些性能优化,并且是跟踪 process.memoryUsage()
中的 ArrayBuffer
内存所必需的。
此外,用于 Node.js 实例的每个 v8::Isolate
都需要在 MultiIsolatePlatform
实例中注册和注销(如果使用),以便平台知道为 v8::Isolate
调度的任务使用哪个事件循环。
node::NewIsolate()
辅助函数创建一个 v8::Isolate
,用一些 Node.js 特定的钩子(例如 Node.js 错误处理程序)设置它,并自动将其注册到平台。
int RunNodeInstance(MultiIsolatePlatform* platform,
const std::vector<std::string>& args,
const std::vector<std::string>& exec_args) {
int exit_code = 0;
// Setup up a libuv event loop, v8::Isolate, and Node.js Environment.
std::vector<std::string> errors;
std::unique_ptr<CommonEnvironmentSetup> setup =
CommonEnvironmentSetup::Create(platform, &errors, args, exec_args);
if (!setup) {
for (const std::string& err : errors)
fprintf(stderr, "%s: %s\n", args[0].c_str(), err.c_str());
return 1;
}
Isolate* isolate = setup->isolate();
Environment* env = setup->env();
{
Locker locker(isolate);
Isolate::Scope isolate_scope(isolate);
HandleScope handle_scope(isolate);
// The v8::Context needs to be entered when node::CreateEnvironment() and
// node::LoadEnvironment() are being called.
Context::Scope context_scope(setup->context());
// Set up the Node.js instance for execution, and run code inside of it.
// There is also a variant that takes a callback and provides it with
// the `require` and `process` objects, so that it can manually compile
// and run scripts as needed.
// The `require` function inside this script does *not* access the file
// system, and can only load built-in Node.js modules.
// `module.createRequire()` is being used to create one that is able to
// load files from the disk, and uses the standard CommonJS file loader
// instead of the internal-only `require` function.
MaybeLocal<Value> loadenv_ret = node::LoadEnvironment(
env,
"const publicRequire ="
" require('node:module').createRequire(process.cwd() + '/');"
"globalThis.require = publicRequire;"
"require('node:vm').runInThisContext(process.argv[1]);");
if (loadenv_ret.IsEmpty()) // There has been a JS exception.
return 1;
exit_code = node::SpinEventLoop(env).FromMaybe(1);
// node::Stop() can be used to explicitly stop the event loop and keep
// further JavaScript from running. It can be called from any thread,
// and will act like worker.terminate() if called from another thread.
node::Stop(env);
}
return exit_code;
}