根据我的经验,php服务器会向日志(log)或服务器端抛出异常,但不会向 node 抛出异常.js只是简单的崩溃.用try-catch来包围我的代码也不起作用,因为一切都是异步完成的.我想知道其他人在他们的生产服务器中做什么.
根据我的经验,php服务器会向日志(log)或服务器端抛出异常,但不会向 node 抛出异常.js只是简单的崩溃.用try-catch来包围我的代码也不起作用,因为一切都是异步完成的.我想知道其他人在他们的生产服务器中做什么.
PM2
首先,我强烈建议为Node.js
安装PM2
.PM2在处理崩溃、监控 node 应用以及负载平衡方面非常出色.每当 node 应用程序崩溃、出于任何原因停止,甚至服务器重新启动时,PM2都会立即启动该应用程序.所以,如果有一天,即使在管理我们的代码之后,应用程序崩溃,PM2也可以立即重启它.欲了解更多信息,请点击Installing and Running PM2
其他答案真的很疯狂,因为你可以在http://nodejs.org/docs/latest/api/process.html#process_event_uncaughtexception点阅读Node自己的文档
如果有人正在使用其他说明的答案,请阅读 node 文档:
请注意,
uncaughtException
是一种非常粗糙的异常处理机制,将来可能会被删除
现在回到我们防止应用程序本身崩溃的解决方案.
因此,经过仔细研究,我最终得出了Node文档本身的建议:
不要用
uncaughtException
,而是用domains
加cluster
.如果使用uncaughtException
,请在每次未处理的异常后重新启动应用程序!
DOMAIN对Cluster
我们实际上要做的是向触发错误的请求发送一个错误响应,同时让其他人在正常时间内完成,并停止侦听该工作进程中的新请求.
通过这种方式,域的使用与集群模块密切相关,因为当工作进程遇到错误时,主进程可以派生新的工作进程.看下面的代码来理解我的意思
通过使用Domain
,以及使用Cluster
将程序划分为多个工作进程的弹性,我们可以做出更恰当的react ,并以更大的安全性处理错误.
var cluster = require('cluster');
var PORT = +process.env.PORT || 1337;
if(cluster.isMaster)
{
cluster.fork();
cluster.fork();
cluster.on('disconnect', function(worker)
{
console.error('disconnect!');
cluster.fork();
});
}
else
{
var domain = require('domain');
var server = require('http').createServer(function(req, res)
{
var d = domain.create();
d.on('error', function(er)
{
//something unexpected occurred
console.error('error', er.stack);
try
{
//make sure we close down within 30 seconds
var killtimer = setTimeout(function()
{
process.exit(1);
}, 30000);
// But don't keep the process open just for that!
killtimer.unref();
//stop taking new requests.
server.close();
//Let the master know we're dead. This will trigger a
//'disconnect' in the cluster master, and then it will fork
//a new worker.
cluster.worker.disconnect();
//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.
d.add(req);
d.add(res);
//Now run the handler function in the domain.
d.run(function()
{
//You'd put your fancy application logic here.
handleRequest(req, res);
});
});
server.listen(PORT);
}
虽然Domain
是待定的弃用,并将被删除,因为新的替代品来了,如 node 的文档中所述
此模块正在等待弃用.一旦完成了替换API,该模块将被完全弃用.绝对必须拥有域提供的功能的用户可能暂时依赖它,但应该期望将来必须迁移到不同的解决方案.
但是,在没有引入新的替代方案之前,具有集群的域是 node 文档所建议的唯一好的解决方案.
要深入理解Domain
和Cluster
,请阅读
https://nodejs.org/api/domain.html#domain_domain (Stability: 0 - Deprecated
)
https://nodejs.org/api/cluster.html
感谢@Stanley Luo与我们分享关于集群和域的精彩深入解释