NodeJS提供了domain
模塊,可以簡化異步代碼的異常處理。在介紹該模塊之前,我們需要首先理解“域”的概念。簡單的講,一個域就是一個JS運行環(huán)境,在一個運行環(huán)境中,如果一個異常沒有被捕獲,將作為一個全局異常被拋出。NodeJS通過process
對象提供了捕獲全局異常的方法,示例代碼如下
process.on('uncaughtException', function (err) {
console.log('Error: %s', err.message);
});
setTimeout(function (fn) {
fn();
});
-- Console ------------------------------
Error: undefined is not a function
雖然全局異常有個地方可以捕獲了,但是對于大多數(shù)異常,我們希望盡早捕獲,并根據(jù)結(jié)果決定代碼的執(zhí)行路徑。我們用以下HTTP服務器代碼作為例子:
function async(request, callback) {
// Do something.
asyncA(request, function (err, data) {
if (err) {
callback(err);
} else {
// Do something
asyncB(request, function (err, data) {
if (err) {
callback(err);
} else {
// Do something
asyncC(request, function (err, data) {
if (err) {
callback(err);
} else {
// Do something
callback(null, data);
}
});
}
});
}
});
}
http.createServer(function (request, response) {
async(request, function (err, data) {
if (err) {
response.writeHead(500);
response.end();
} else {
response.writeHead(200);
response.end(data);
}
});
});
以上代碼將請求對象交給異步函數(shù)處理后,再根據(jù)處理結(jié)果返回響應。這里采用了使用回調(diào)函數(shù)傳遞異常的方案,因此async
函數(shù)內(nèi)部如果再多幾個異步函數(shù)調(diào)用的話,代碼就變成上邊這副鬼樣子了。為了讓代碼好看點,我們可以在每處理一個請求時,使用domain
模塊創(chuàng)建一個子域(JS子運行環(huán)境)。在子域內(nèi)運行的代碼可以隨意拋出異常,而這些異??梢酝ㄟ^子域?qū)ο蟮?code>error事件統(tǒng)一捕獲。于是以上代碼可以做如下改造:
function async(request, callback) {
// Do something.
asyncA(request, function (data) {
// Do something
asyncB(request, function (data) {
// Do something
asyncC(request, function (data) {
// Do something
callback(data);
});
});
});
}
http.createServer(function (request, response) {
var d = domain.create();
d.on('error', function () {
response.writeHead(500);
response.end();
});
d.run(function () {
async(request, function (data) {
response.writeHead(200);
response.end(data);
});
});
});
可以看到,我們使用.create
方法創(chuàng)建了一個子域?qū)ο?,并通過.run
方法進入需要在子域中運行的代碼的入口點。而位于子域中的異步函數(shù)回調(diào)函數(shù)由于不再需要捕獲異常,代碼一下子瘦身很多。
無論是通過process
對象的uncaughtException
事件捕獲到全局異常,還是通過子域?qū)ο蟮?code>error事件捕獲到了子域異常,在NodeJS官方文檔里都強烈建議處理完異常后立即重啟程序,而不是讓程序繼續(xù)運行。按照官方文檔的說法,發(fā)生異常后的程序處于一個不確定的運行狀態(tài),如果不立即退出的話,程序可能會發(fā)生嚴重內(nèi)存泄漏,也可能表現(xiàn)得很奇怪。
但這里需要澄清一些事實。JS本身的throw..try..catch
異常處理機制并不會導致內(nèi)存泄漏,也不會讓程序的執(zhí)行結(jié)果出乎意料,但NodeJS并不是存粹的JS。NodeJS里大量的API內(nèi)部是用C/C++實現(xiàn)的,因此NodeJS程序的運行過程中,代碼執(zhí)行路徑穿梭于JS引擎內(nèi)部和外部,而JS的異常拋出機制可能會打斷正常的代碼執(zhí)行流程,導致C/C++部分的代碼表現(xiàn)異常,進而導致內(nèi)存泄漏等問題。
因此,使用uncaughtException
或domain
捕獲異常,代碼執(zhí)行路徑里涉及到了C/C++部分的代碼時,如果不能確定是否會導致內(nèi)存泄漏等問題,最好在處理完異常后重啟程序比較妥當。而使用try
語句捕獲異常時一般捕獲到的都是JS本身的異常,不用擔心上訴問題。
更多建議: