1 <div class="line number2 index1 alt1">
try {
</div>
<div class="line number3 index2 alt2">
process.nextTick(function () {
</div>
<div class="line number4 index3 alt1">
throw new Error("error");
</div>
<div class="line number5 index4 alt2">
});
</div>
<div class="line number6 index5 alt1">
} catch (err) {
</div>
<div class="line number7 index6 alt2">
//can not catch it
</div>
<div class="line number8 index7 alt1">
console.log(err);
</div>
<div class="line number9 index8 alt2">
}
</div>
<div class="line number10 index9 alt1">
</div>
<div class="line number11 index10 alt2">
</div>
<div class="line number12 index11 alt1">
try {
</div>
<div class="line number13 index12 alt2">
setTimeout(function(){
</div>
<div class="line number14 index13 alt1">
throw new Error("error");
</div>
<div class="line number15 index14 alt2">
},1)
</div>
<div class="line number16 index15 alt1">
} catch (err) {
</div>
<div class="line number17 index16 alt2">
//can not catch it
</div>
<div class="line number18 index17 alt1">
console.log(err);
</div>
<div class="line number19 index18 alt2">
}
</div>
</div>
</td>
</tr>
</table>
上面的代码没有像预期的那样帮我们捕获异常,后果就是这个未被捕获的异常导致整个Node进程crash,所以Node中try/catch的方式不是那么管用。 如果有一种方法能帮我们全局捕获异常,Node服务就不会轻易挂掉了。Node确实提供了这种方式,不过却不能完全满足我们的需求。 1.2 uncaughtException #当一个异常未被捕获,冒泡回归到事件循环中就会触发uncaughtException事件。 1 <div class="line number2 index1 alt1">
process.on('uncaughtException', function(err) {
</div>
<div class="line number3 index2 alt2">
console.error('Error caught in uncaughtException event:', err);
</div>
<div class="line number4 index3 alt1">
});
</div>
<div class="line number5 index4 alt2">
</div>
<div class="line number6 index5 alt1">
</div>
<div class="line number7 index6 alt2">
try {
</div>
<div class="line number8 index7 alt1">
setTimeout(function(){
</div>
<div class="line number9 index8 alt2">
throw new Error("error");
</div>
<div class="line number10 index9 alt1">
},1)
</div>
<div class="line number11 index10 alt2">
} catch (err) {
</div>
<div class="line number12 index11 alt1">
//can not catch it
</div>
<div class="line number13 index12 alt2">
console.log(err);
</div>
<div class="line number14 index13 alt1">
}
</div>
</div>
</td>
</tr>
</table>
只要给uncaughtException配置了回调,Node进程不会异常退出,但异常发生的上下文已经丢失,我们无法给出友好的返回,比如告诉用户哪里出问题了。而且由于uncaughtException事件发生后,会丢失当前环境的堆栈,可能导致Node不能正常进行内存回收,从而导致内存泄露。所以,uncaughtException的正确使用姿势是,当uncaughtException触发,记录error日志,然后结束Node进程,我们通过日志监控,报警及时解决异常。 1 <div class="line number2 index1 alt1">
process.on('uncaughtException', function(err) {
</div>
<div class="line number3 index2 alt2">
// 记录日志
</div>
<div class="line number4 index3 alt1">
logger(err);
</div>
<div class="line number5 index4 alt2">
// 结束进程
</div>
<div class="line number6 index5 alt1">
process.exit(1);
</div>
<div class="line number7 index6 alt2">
});
</div>
</div>
</td>
</tr>
</table>
1.3 domain #为了弥补try/catch和uncaughtException的不足,在node v0.8+版本的时候,发布了一个模块domain ,这个模块能捕捉异步回调中出现的异常。 看下面的示例: 1 <div class="line number2 index1 alt1">
var d = domain.create();
</div>
<div class="line number3 index2 alt2">
</div>
<div class="line number4 index3 alt1">
process.on('uncaughtException', function(err) {
</div>
<div class="line number5 index4 alt2">
console.error(err);
</div>
<div class="line number6 index5 alt1">
});
</div>
<div class="line number7 index6 alt2">
</div>
<div class="line number8 index7 alt1">
d.on('error', function(err) {
</div>
<div class="line number9 index8 alt2">
console.error('Error caught by domain:', err);
</div>
<div class="line number10 index9 alt1">
});
</div>
<div class="line number11 index10 alt2">
</div>
<div class="line number12 index11 alt1">
d.run(function() {
</div>
<div class="line number13 index12 alt2">
process.nextTick(function() {
</div>
<div class="line number14 index13 alt1">
throw new Error("test domain");
</div>
<div class="line number15 index14 alt2">
});
</div>
<div class="line number16 index15 alt1">
});
</div>
</div>
</td>
</tr>
</table>
运行代码我们会发现,异常会被domain捕获到,uncaughtException不会被触发。虽然我们对domain模块寄予厚望,不过目前domain模块的评级为“Unstable”,因为存在不少性能以及稳定性问题。 关于domain的详细介绍,可以看看以下几篇文章: 二 阻止异常发生 #我们当然无法阻止异常的发生,这里说的阻止异常发生,是希望大家能养成良好的编码习惯,严谨的思维逻辑,来尽可能减少代码抛出未捕获异常。 2.1 良好的异常处理习惯 #- 异步API编写规范:由于异步调用中回调函数里的异常无法被外部捕获,所以我们将API内部发生的异常作为第一个参数传递给回调函数,包括NodeJS官方的API是遵循这个规范的。
1 <div class="line number2 index1 alt1">
fs.readFile('/t.txt', function (err, data) {
</div>
<div class="line number3 index2 alt2">
if (err) throw err;
</div>
<div class="line number4 index3 alt1">
console.log(data);
</div>
<div class="line number5 index4 alt2">
});
</div>
<div class="line number6 index5 alt1">
</div>
<div class="line number7 index6 alt2">
</div>
<div class="line number8 index7 alt1">
// 不推荐的做法
</div>
<div class="line number9 index8 alt2">
function fun(options,callback){
</div>
<div class="line number10 index9 alt1">
if(!options{
</div>
<div class="line number11 index10 alt2">
throw new Error("..")
</div>
<div class="line number12 index11 alt1">
}
</div>
<div class="line number13 index12 alt2">
}
</div>
<div class="line number14 index13 alt1">
</div>
<div class="line number15 index14 alt2">
// 推荐的做法
</div>
<div class="line number16 index15 alt1">
function fun(options,callback){
</div>
<div class="line number17 index16 alt2">
if(!options){
</div>
<div class="line number18 index17 alt1">
callback(err,null)
</div>
<div class="line number19 index18 alt2">
}
</div>
<div class="line number20 index19 alt1">
}
</div>
</div>
</td>
</tr>
</table>
1 <div class="line number2 index1 alt1">
function doSth(cb){
</div>
<div class="line number3 index2 alt2">
if(typeof cb === 'function'){
</div>
<div class="line number4 index3 alt1">
cb();
</div>
<div class="line number5 index4 alt2">
}
</div>
<div class="line number6 index5 alt1">
}
</div>
</div>
</td>
</tr>
</table>
1 <div class="line number2 index1 alt1">
var obj;
</div>
<div class="line number3 index2 alt2">
try{
</div>
<div class="line number4 index3 alt1">
obj = JSON.parse('')
</div>
<div class="line number5 index4 alt2">
}catch(e){
</div>
<div class="line number6 index5 alt1">
obj = {};
</div>
<div class="line number7 index6 alt2">
}
</div>
</div>
</td>
</tr>
</table>
- 不要直接在controller中抛异常,应该用500等状态更友好的返回错误
1 <div class="line number2 index1 alt1">
// 不推荐的做法
</div>
<div class="line number3 index2 alt2">
app.get('/item.html', function (req, res, next) {
</div>
<div class="line number4 index3 alt1">
if(!query["id"]){
</div>
<div class="line number5 index4 alt2">
throw new Error('no item id');
</div>
<div class="line number6 index5 alt1">
}
</div>
<div class="line number7 index6 alt2">
});
</div>
<div class="line number8 index7 alt1">
</div>
<div class="line number9 index8 alt2">
// 推荐的做法
</div>
<div class="line number10 index9 alt1">
app.get('/item.html', function (req, res, next) {
</div>
<div class="line number11 index10 alt2">
if(!query["id"]){
</div>
<div class="line number12 index11 alt1">
next();
</div>
<div class="line number13 index12 alt2">
}
</div>
<div class="line number14 index13 alt1">
});
</div>
</div>
</td>
</tr>
</table>
2.2 单元测试 #单元测试的重要性想必所有人都清楚,JS代码跑在浏览器端的时候我们未必会做,因为异常通常只会影响部分人使用的部分功能,不足以引起很多人的重视。JS代码跑在服务端情况完全不一样了,一旦出现异常,影响的是所有的用户,所以单元测试就显得非常重要。 至于如何在NodeJS中写单元测试,可以看看两位大神的分享: 2.3 记录日志 #还是那句说,谁也不敢保证自己的代码不出现异常,因为有运行环境,网络环境等各种不稳定因素,所以建立健全的排查和跟踪机制就显得很重要,而日志就是实现这种机制的关键。 阿里线上环境已经有完善的日志监控体现,我们要做的就是去学会如何使用他。 ali-logger:http://search.npm.taobao.net/package/ali-logger 三 多进程架构 #3.1 cluster #cluster模块用于创建共享端口的多进程模式,这种模式使多个进程间共享一个监听状态的socket,并由系统将accept的connection分配给不同的子进程。 文档上有一个简单的示例: 1 <div class="line number2 index1 alt1">
var cluster = require('cluster');
</div>
<div class="line number3 index2 alt2">
var http = require('http');
</div>
<div class="line number4 index3 alt1">
var numCPUs = require('os').cpus().length;
</div>
<div class="line number5 index4 alt2">
</div>
<div class="line number6 index5 alt1">
if (cluster.isMaster) {
</div>
<div class="line number7 index6 alt2">
// Fork workers.
</div>
<div class="line number8 index7 alt1">
for (var i = 0; i < numCPUs; i++) {
</div>
<div class="line number9 index8 alt2">
cluster.fork();
</div>
<div class="line number10 index9 alt1">
}
</div>
<div class="line number11 index10 alt2">
</div>
<div class="line number12 index11 alt1">
cluster.on('exit', function(worker, code, signal) {
</div>
<div class="line number13 index12 alt2">
console.log('worker ' + worker.process.pid + ' died');
</div>
<div class="line number14 index13 alt1">
});
</div>
<div class="line number15 index14 alt2">
} else {
</div>
<div class="line number16 index15 alt1">
// Workers can share any TCP connection
</div>
<div class="line number17 index16 alt2">
// In this case its a HTTP server
</div>
<div class="line number18 index17 alt1">
http.createServer(function(req, res) {
</div>
<div class="line number19 index18 alt2">
res.writeHead(200);
</div>
<div class="line number20 index19 alt1">
res.end("hello world\n");
</div>
<div class="line number21 index20 alt2">
}).listen(8000);
</div>
<div class="line number22 index21 alt1">
}
</div>
</div>
</td>
</tr>
</table>
利用cluster ,我们可以根据CPU的数量创建多个worker进程,用户的请求会被分配到不同的进程上,如果某个进程出现异常,可以直接将这个进程crash掉,而不会影响其他进程。 关于cluster 实现原理的介绍文章非常多,想深入了解的可以自行搜索一下。 3.1.1 graceful + recluster {#3-1-1_graceful_+_recluster} #数据产品中使用graceful + recluster两个模块实现多进程和服务器稳定性的工作,分享一下使用方法: app.js 1 <div class="line number2 index1 alt1">
var app = require('../app');
</div>
<div class="line number3 index2 alt2">
var graceful = require('graceful');
</div>
<div class="line number4 index3 alt1">
</div>
<div class="line number5 index4 alt2">
var server = app.listen(app.get('port'), function() {
</div>
<div class="line number6 index5 alt1">
debug('Express server listening on port ' + server.address().port);
</div>
<div class="line number7 index6 alt2">
});
</div>
<div class="line number8 index7 alt1">
</div>
<div class="line number9 index8 alt2">
graceful({
</div>
<div class="line number10 index9 alt1">
server: server,
</div>
<div class="line number11 index10 alt2">
killTimeout: 30000,
</div>
<div class="line number12 index11 alt1">
error:function(e){
</div>
<div class="line number13 index12 alt2">
logger.error(e);
</div>
<div class="line number14 index13 alt1">
}
</div>
<div class="line number15 index14 alt2">
});
</div>
</div>
</td>
</tr>
</table>
server.js var recluster = require('recluster'), <div class="line number2 index1 alt1">
path = require('path');
</div>
<div class="line number3 index2 alt2">
</div>
<div class="line number4 index3 alt1">
var cluster = recluster(path.join(__dirname, 'app.js'));
</div>
<div class="line number5 index4 alt2">
cluster.run();
</div>
<div class="line number6 index5 alt1">
</div>
<div class="line number7 index6 alt2">
process.on('SIGUSR2', function() {
</div>
<div class="line number8 index7 alt1">
console.log('Got SIGUSR2, reloading cluster...');
</div>
<div class="line number9 index8 alt2">
cluster.reload();
</div>
<div class="line number10 index9 alt1">
});
</div>
<div class="line number11 index10 alt2">
</div>
<div class="line number12 index11 alt1">
console.log("spawned cluster, kill -s SIGUSR2", process.pid, "to reload");
</div>
</div>
</td>
</tr>
</table>
开发环境,直接通过app启动: 生产环境, 启动多个 worker: 3.1.2 pm2 #如果你觉得graceful + recluster的方式太复杂,那么pm2肯定是你最理想的选择。 pm2非常强大,生产环境使用pm2启动你的Node服务是个不错的选择,他能自动利用你的多核cup,完善的监控,日志记录等等.. pm2使用非常简单: $ npm install pm2@latest -g <div class="line number2 index1 alt1">
</div>
<div class="line number3 index2 alt2">
$ pm2 start app.js
</div>
<div class="line number4 index3 alt1">
</div>
<div class="line number5 index4 alt2">
$ pm2 list
</div>
</div>
</td>
</tr>
</table>

如果你想更多的了解pm2,可以直接看pm2的文档:https://github.com/Unitech/pm2 不过正是由于pm2功能太过强大,我们没有选择pm2,因为他太复杂了,感觉还没有能力驾驭他,特别是万一出现问题,我们还不知道如何解决。也许我们的顾虑是多余的,不过需要一点时间去了解他。 小结 #刚开始学习Node的使用,对Node确实有点小担心,因为使用别的语言做Web服务,根本不用担心因为一个错误导致整个服务crash的问题。 随着对Node的了解,我掌握了一些技巧,也打消了一些顾虑。 不过毕竟Node应用经验有限,所以欢迎大家一起探讨,积累更多宝贵的经验。 转载:https://www.cnblogs.com/rubylouvre/p/4442619.html
请遵守《互联网环境法规》文明发言,欢迎讨论问题
|
|
|
|
|
|
|
|
|
|
|
|