Node.js v25.0.0 文档
- Node.js v25.0.0
- 目录
-
索引
- 断言测试
- 异步上下文跟踪
- 异步钩子
- 缓冲区
- C++ 插件
- 使用 Node-API 的 C/C++ 插件
- C++ 嵌入器 API
- 子进程
- 集群
- 命令行选项
- 控制台
- 加密
- 调试器
- 已弃用的 API
- 诊断通道
- DNS
- 域
- 环境变量
- 错误
- 事件
- 文件系统
- 全局对象
- HTTP
- HTTP/2
- HTTPS
- 检查器
- 国际化
- 模块:CommonJS 模块
- 模块:ECMAScript 模块
- 模块:
node:module
API - 模块:包
- 模块:TypeScript
- 网络
- 操作系统
- 路径
- 性能钩子
- 权限
- 进程
- Punycode
- 查询字符串
- 逐行读取
- REPL
- 报告
- 单一可执行文件应用
- SQLite
- 流
- 字符串解码器
- 测试运行器
- 定时器
- TLS/SSL
- 跟踪事件
- TTY
- UDP/数据报
- URL
- 实用工具
- V8
- 虚拟机
- WASI
- Web Crypto API
- Web Streams API
- 工作线程
- Zlib
- 其他版本
- 选项
域#
源代码: lib/domain.js
该模块即将被弃用。 一旦替代 API 最终确定,该模块将被完全弃用。大多数开发者 不 应该有理由使用这个模块。对于绝对必须使用域所提供功能的用户,可以暂时依赖它,但应预期未来需要迁移到不同的解决方案。
域提供了一种将多个不同的 IO 操作作为一个单独的组来处理的方法。如果任何注册到域的事件发射器或回调函数触发了一个 'error'
事件,或者抛出了一个错误,那么域对象将会收到通知,而不是在 process.on('uncaughtException')
处理程序中丢失错误的上下文,或者导致程序立即以一个错误码退出。
警告:不要忽略错误!#
域错误处理程序不能替代在发生错误时关闭进程。
根据 JavaScript 中 throw
的工作原理,几乎没有任何方法可以安全地“从中断的地方继续”,而不会泄漏引用,或造成其他某种未定义的脆弱状态。
响应抛出错误的最安全方式是关闭进程。当然,在一个正常的 Web 服务器中,可能有很多打开的连接,因为别人触发的一个错误而突然关闭这些连接是不合理的。
更好的方法是向触发错误的请求发送一个错误响应,同时让其他请求在正常时间内完成,并停止在该工作进程中监听新的请求。
通过这种方式,domain
的使用与 cluster 模块紧密相连,因为主进程可以在一个工作进程遇到错误时 fork 一个新的工作进程。对于扩展到多台机器的 Node.js 程序,终止代理或服务注册中心可以注意到这个故障,并作出相应的反应。
例如,这不是一个好主意
// XXX WARNING! BAD IDEA!
const d = require('node:domain').create();
d.on('error', (er) => {
// The error won't crash the process, but what it does is worse!
// Though we've prevented abrupt process restarting, we are leaking
// a lot of resources if this ever happens.
// This is no better than process.on('uncaughtException')!
console.log(`error, but oh well ${er.message}`);
});
d.run(() => {
require('node:http').createServer((req, res) => {
handleRequest(req, res);
}).listen(PORT);
});
通过使用域的上下文,以及将我们的程序分离到多个工作进程中的弹性,我们可以更恰当地作出反应,并以更高的安全性处理错误。
// Much better!
const cluster = require('node:cluster');
const PORT = +process.env.PORT || 1337;
if (cluster.isPrimary) {
// A more realistic scenario would have more than 2 workers,
// and perhaps not put the primary and worker in the same file.
//
// It is also possible to get a bit fancier about logging, and
// implement whatever custom logic is needed to prevent DoS
// attacks and other bad behavior.
//
// See the options in the cluster documentation.
//
// The important thing is that the primary does very little,
// increasing our resilience to unexpected errors.
cluster.fork();
cluster.fork();
cluster.on('disconnect', (worker) => {
console.error('disconnect!');
cluster.fork();
});
} else {
// the worker
//
// This is where we put our bugs!
const domain = require('node:domain');
// See the cluster documentation for more details about using
// worker processes to serve requests. How it works, caveats, etc.
const server = require('node:http').createServer((req, res) => {
const d = domain.create();
d.on('error', (er) => {
console.error(`error ${er.stack}`);
// We're in dangerous territory!
// By definition, something unexpected occurred,
// which we probably didn't want.
// Anything can happen now! Be very careful!
try {
// Make sure we close down within 30 seconds
const killtimer = setTimeout(() => {
process.exit(1);
}, 30000);
// But don't keep the process open just for that!
killtimer.unref();
// Stop taking new requests.
server.close();
// Let the primary know we're dead. This will trigger a
// 'disconnect' in the cluster primary, and then it will fork
// a new worker.
cluster.worker.disconnect();
// Try to send an error to the request that triggered the problem
res.statusCode = 500;
res.setHeader('content-type', 'text/plain');
res.end('Oops, there was a problem!\n');
} catch (er2) {
// Oh well, not much we can do at this point.
console.error(`Error sending 500! ${er2.stack}`);
}
});
// Because req and res were created before this domain existed,
// we need to explicitly add them.
// See the explanation of implicit vs explicit binding below.
d.add(req);
d.add(res);
// Now run the handler function in the domain.
d.run(() => {
handleRequest(req, res);
});
});
server.listen(PORT);
}
// This part is not important. Just an example routing thing.
// Put fancy application logic here.
function handleRequest(req, res) {
switch (req.url) {
case '/error':
// We do some async stuff, and then...
setTimeout(() => {
// Whoops!
flerb.bark();
}, timeout);
break;
default:
res.end('ok');
}
}
对 Error
对象的补充#
任何时候一个 Error
对象通过一个域进行路由时,都会给它添加一些额外的字段。
error.domain
第一个处理该错误的域。error.domainEmitter
触发了带有该错误对象的'error'
事件的事件发射器。error.domainBound
绑定到该域的回调函数,并且错误作为其第一个参数被传入。error.domainThrown
一个布尔值,指示错误是被抛出、触发,还是传递给一个绑定的回调函数。
隐式绑定#
如果正在使用域,那么所有 新 的 EventEmitter
对象(包括 Stream 对象、请求、响应等)将在其创建时隐式地绑定到活动域。
此外,传递给低级事件循环请求(例如 fs.open()
或其他接受回调的方法)的回调函数将自动绑定到活动域。如果它们抛出错误,那么域将捕获该错误。
为了防止过度的内存使用,Domain
对象本身不会作为活动域的子对象被隐式添加。如果这样做,将太容易导致请求和响应对象无法被正确地垃圾回收。
要将 Domain
对象作为父 Domain
的子对象进行嵌套,必须显式地添加它们。
隐式绑定会将抛出的错误和 'error'
事件路由到 Domain
的 'error'
事件,但不会在 Domain
上注册 EventEmitter
。隐式绑定只处理抛出的错误和 'error'
事件。
显式绑定#
有时,正在使用的域可能不是应该用于特定事件发射器的域。或者,事件发射器可能在一个域的上下文中创建,但应该被绑定到某个其他的域。
例如,可能有一个用于 HTTP 服务器的域,但我们可能希望为每个请求使用一个单独的域。
这可以通过显式绑定来实现。
// Create a top-level domain for the server
const domain = require('node:domain');
const http = require('node:http');
const serverDomain = domain.create();
serverDomain.run(() => {
// Server is created in the scope of serverDomain
http.createServer((req, res) => {
// Req and res are also created in the scope of serverDomain
// however, we'd prefer to have a separate domain for each request.
// create it first thing, and add req and res to it.
const reqd = domain.create();
reqd.add(req);
reqd.add(res);
reqd.on('error', (er) => {
console.error('Error', er, req.url);
try {
res.writeHead(500);
res.end('Error occurred, sorry.');
} catch (er2) {
console.error('Error sending 500', er2, req.url);
}
});
}).listen(1337);
});
domain.create()
#
- 返回: <Domain>
类: Domain
#
- 继承自:<EventEmitter>
Domain
类封装了将错误和未捕获的异常路由到活动 Domain
对象的功能。
要处理它捕获的错误,请监听其 'error'
事件。
domain.add(emitter)
#
emitter
<EventEmitter> | <Timer> 要添加到域的发射器或定时器
显式地将一个发射器添加到域。如果该发射器调用的任何事件处理程序抛出错误,或者如果该发射器触发了一个 'error'
事件,它将被路由到域的 'error'
事件,就像隐式绑定一样。
这也适用于从 setInterval()
和 setTimeout()
返回的定时器。如果它们的回调函数抛出错误,它将被域的 'error'
处理程序捕获。
如果该 Timer 或 EventEmitter
已经绑定到一个域,它会从那个域中移除,然后绑定到这个域。
domain.bind(callback)
#
callback
<Function> 回调函数- 返回: <Function> 绑定的函数
返回的函数将是所提供回调函数的一个包装器。当调用返回的函数时,任何抛出的错误都将被路由到域的 'error'
事件。
const d = domain.create();
function readSomeFile(filename, cb) {
fs.readFile(filename, 'utf8', d.bind((er, data) => {
// If this throws, it will also be passed to the domain.
return cb(er, data ? JSON.parse(data) : null);
}));
}
d.on('error', (er) => {
// An error occurred somewhere. If we throw it now, it will crash the program
// with the normal line number and stack message.
});
domain.enter()
#
enter()
方法是 run()
、bind()
和 intercept()
方法用于设置活动域的底层机制。它将 domain.active
和 process.domain
设置为该域,并隐式地将该域推入由 domain 模块管理的域堆栈(有关域堆栈的详细信息,请参见 domain.exit()
)。对 enter()
的调用界定了一个绑定到域的异步调用和 I/O 操作链的开始。
调用 enter()
只改变活动域,而不会改变域本身。enter()
和 exit()
可以在单个域上调用任意次数。
domain.exit()
#
exit()
方法退出当前域,将其从域堆栈中弹出。任何时候当执行将要切换到不同异步调用链的上下文时,确保退出当前域是很重要的。对 exit()
的调用界定了绑定到域的异步调用和 I/O 操作链的结束或中断。
如果当前执行上下文绑定了多个嵌套的域,exit()
将退出该域内嵌套的任何域。
调用 exit()
只改变活动域,而不会改变域本身。enter()
和 exit()
可以在单个域上调用任意次数。
domain.intercept(callback)
#
callback
<Function> 回调函数- 返回: <Function> 被拦截的函数
这个方法几乎与 domain.bind(callback)
完全相同。但是,除了捕获抛出的错误外,它还会拦截作为函数的第一个参数发送的 Error
对象。
通过这种方式,常见的 if (err) return callback(err);
模式可以在一个地方用一个单一的错误处理程序来代替。
const d = domain.create();
function readSomeFile(filename, cb) {
fs.readFile(filename, 'utf8', d.intercept((data) => {
// Note, the first argument is never passed to the
// callback since it is assumed to be the 'Error' argument
// and thus intercepted by the domain.
// If this throws, it will also be passed to the domain
// so the error-handling logic can be moved to the 'error'
// event on the domain instead of being repeated throughout
// the program.
return cb(null, JSON.parse(data));
}));
}
d.on('error', (er) => {
// An error occurred somewhere. If we throw it now, it will crash the program
// with the normal line number and stack message.
});
domain.remove(emitter)
#
emitter
<EventEmitter> | <Timer> 要从域中移除的发射器或定时器
与 domain.add(emitter)
相反。从指定的发射器中移除域处理。
domain.run(fn[, ...args])
#
fn
<Function>...args
<any>
在域的上下文中运行提供的函数,隐式地绑定在该上下文中创建的所有事件发射器、定时器和低级请求。可以选择性地将参数传递给该函数。
这是使用域最基本的方式。
const domain = require('node:domain');
const fs = require('node:fs');
const d = domain.create();
d.on('error', (er) => {
console.error('Caught error!', er);
});
d.run(() => {
process.nextTick(() => {
setTimeout(() => { // Simulating some various async stuff
fs.open('non-existent file', 'r', (er, fd) => {
if (er) throw er;
// proceed...
});
}, 100);
});
});
在这个例子中,d.on('error')
处理程序将被触发,而不是使程序崩溃。
域和 Promise#
自 Node.js 8.0.0 起,Promise 的处理程序在调用 .then()
或 .catch()
本身所在的域内运行。
const d1 = domain.create();
const d2 = domain.create();
let p;
d1.run(() => {
p = Promise.resolve(42);
});
d2.run(() => {
p.then((v) => {
// running in d2
});
});
回调函数可以使用 domain.bind(callback)
绑定到特定的域。
const d1 = domain.create();
const d2 = domain.create();
let p;
d1.run(() => {
p = Promise.resolve(42);
});
d2.run(() => {
p.then(p.domain.bind((v) => {
// running in d1
}));
});
域不会干扰 Promise 的错误处理机制。换句话说,对于未处理的 Promise
拒绝,不会触发 'error'
事件。