在网上看到一些帖子,吐糟,质疑nodejs 程序的稳定性,为何呢?其一,可能这个和javascript有关吧,node是拿javascript去实现的,而javascript又被称为是“世界上误解最深的语言”,咱们能够去看看nodejs 创始人的说法,能够去看看知乎的这篇文章,为何node 用javascript去实现,其二,nodejs 毕竟还年轻,并且官网在部分模块也标注了此模块的目前的状态。javascript
这2天我抽了点时间思考了下这个问题,我以为咱们程序首先功能应该是独立的,就是一个功能出异常了,不该该去影响另一个正常的功能,不该该将整个程序都崩溃掉,其次,即便是程序崩溃了,咱们也应该有一个让程序自动启动,另外,应该去记录日志,方便咱们跟踪问题。我以为主要能够从如下方面提升nodejs 稳定性:html
1)保持良好的代码结构:java
咱们知道node是单线程,非阻塞io,默认就是异步,经过回调的方式处理后面的流程,若是嵌套的层次太多了,势必会引发代码逻辑结构的混乱,也不利于维护和升级,能够采用async这个异步流程控制模块,来理清咱们的代码逻辑。node
2)使用 process.on('uncaughtException', function(err){...}); 来处理未被捕捉的错误。git
3)使用try~catch 来捕获异常:github
这个只能解决一部分问题,不是万能的,在上面说到由于node是单线程,非阻塞io,默认就是异步,经过回调的方式处理后面的流程,try~catch 是不能捕获的callback 里面的error的错误的,怎么捕获到callback里面的错误呢 ? 能够采用domain模块api
4)使用domain模块来处理程序的异常服务器
先看看对domain的解释:domain是 EventEmitter类的一个子类。监听它的error
事件来处理它捕捉到的错误。 它提供了一种方式,即以一个单一的组的形式来处理多个不一样的IO操做。若是任何一个注册到domain的事件触发器或回调触发了一个‘error’事件,或者抛出一个错误,那么domain对象将会被通知到。而不是直接让这个错误的上下文从`process.on('uncaughtException')'处理程序中丢失掉,也不会导致程序由于这个错误伴随着错误码当即退出。app
如何使用domain 模块呢?看一个例子:dom
serverDomain.run(function() { // 服务器在serverDomain的做用域内被建立 http.createServer(function(req, res) { // req和res一样在serverDomain的做用域内被建立 // 可是,咱们想对于每个请求使用一个不同的域。 // 因此咱们首先建立一个域,而后将req和res添加到这个域上。 var reqd = domain.create(); reqd.add(req); reqd.add(res); reqd.on('error', function(er) { console.error('Error', er, req.url); try { res.writeHead(500); res.end('Error occurred, sorry.'); } catch (er) { console.error('Error sending 500', er, req.url); } }); }).listen(1337); }); ```
说明:首先建立一个域(domain.create()),而后将须要监控的分发器添加到该域上,最后给域绑定一个错误事件,这样就能够监控了。
再看一个例子:
var d = domain.create(); d.on('error', function(er) { console.error('Caught error!', er); }); d.run(function() { process.nextTick(function() { setTimeout(function() { // 模拟几个不一样的异步的东西 fs.open('non-existent file', 'r', function(er, fd) { if (er) throw er; // 继续。。。 }); }, 100); }); });
说明:首先建立一个域,给域绑定一个错误事件,而后在域的上下文提供能够运行的函数
若是对于回调呢?能够这么使用
var d = domain.create(); function readSomeFile(filename, cb) { fs.readFile(filename, 'utf8', d.bind(function(er, data) { // if this throws, it will also be passed to the domain return cb(er, data ? JSON.parse(data) : null); })); } d.on('error', function(er) { // an error occurred somewhere. // if we throw it now, it will crash the program // with the normal line number and stack message. });
固然也能够这么使用
var d = domain.create(); function readSomeFile(filename, cb) { fs.readFile(filename, 'utf8', d.intercept(function(data) { return cb(null, JSON.parse(data)); })); } d.on('error', function(er) { // an error occurred somewhere. // if we throw it now, it will crash the program // with the normal line number and stack message });
这个函数与domain.bind(callback)
几乎如出一辙。可是,除了捕捉被抛出的错误外,它还会拦截做为第一参数被传递到这个函数的Error
对象。
5)使用log4js 模块记录日志
log4js 是一个很是强大的日志管理工具,在能够看看github这个项目: https://github.com/nomiddlename/log4js-node
6)使用forever 模块来管理nodejs
forever 是服务端管理nodejs 的一个模块,一个命令行工具,可以启动,中止app 应用。forever彻底是基于命令行操做,在forever进程管理之下,建立node的子进程,经过monitor监控node子进程的运行状况,一旦文件更新,或者进程挂掉,forever会自动重启node服务器,确保应用正常运行。很是的好用.
能够关注下这个项目:https://github.com/nodejitsu/forever
可是forever 也不是万能的,也由下面这些问题:
附本文测试代码:https://github.com/yupeng528/node-error