国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 開發(fā) > JS > 正文

解析Node.js異常處理中domain模塊的使用方法

2024-05-06 16:28:27
字體:
供稿:網(wǎng)友
這篇文章主要介紹了Node.js異常處理中domain模塊的使用方法,文中最后提到了內(nèi)存泄漏的相關(guān)問題,值得注意,需要的朋友可以參考下
 

NodeJS 提供了 domain 模塊,可以簡(jiǎn)化異步代碼的異常處理。在介紹該模塊之前,我們需要首先理解“域”的概念。簡(jiǎn)單的講,一個(gè)域就是一個(gè) JS 運(yùn)行環(huán)境,在一個(gè)運(yùn)行環(huán)境中,如果一個(gè)異常沒有被捕獲,將作為一個(gè)全局異常被拋出。NodeJS 通過 process 對(duì)象提供了捕獲全局異常的方法,示例代碼如下

process.on('uncaughtException', function (err) {  console.log('Error: %s', err.message);});setTimeout(function (fn) {  fn();});
Error: undefined is not a function

雖然全局異常有個(gè)地方可以捕獲了,但是對(duì)于大多數(shù)異常,我們希望盡早捕獲,并根據(jù)結(jié)果決定代碼的執(zhí)行路徑。我們用以下 HTTP 服務(wù)器代碼作為例子:

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);    }  });});

以上代碼將請(qǐng)求對(duì)象交給異步函數(shù)處理后,再根據(jù)處理結(jié)果返回響應(yīng)。這里采用了使用回調(diào)函數(shù)傳遞異常的方案,因此 async 函數(shù)內(nèi)部如果再多幾個(gè)異步函數(shù)調(diào)用的話,代碼就變成上邊這副鬼樣子了。為了讓代碼好看點(diǎn),我們可以在每處理一個(gè)請(qǐng)求時(shí),使用 domain 模塊創(chuàng)建一個(gè)子域(JS 子運(yùn)行環(huán)境)。在子域內(nèi)運(yùn)行的代碼可以隨意拋出異常,而這些異常可以通過子域?qū)ο蟮?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)建了一個(gè)子域?qū)ο螅⑼ㄟ^.run方法進(jìn)入需要在子域中運(yùn)行的代碼的入口點(diǎn)。而位于子域中的異步函數(shù)回調(diào)函數(shù)由于不再需要捕獲異常,代碼一下子瘦身很多。

陷阱
無論是通過 process 對(duì)象的 uncaughtException 事件捕獲到全局異常,還是通過子域?qū)ο蟮?error 事件捕獲到了子域異常,在 NodeJS 官方文檔里都強(qiáng)烈建議處理完異常后立即重啟程序,而不是讓程序繼續(xù)運(yùn)行。按照官方文檔的說法,發(fā)生異常后的程序處于一個(gè)不確定的運(yùn)行狀態(tài),如果不立即退出的話,程序可能會(huì)發(fā)生嚴(yán)重內(nèi)存泄漏,也可能表現(xiàn)得很奇怪。

但這里需要澄清一些事實(shí)。JS 本身的throw..try..catch異常處理機(jī)制并不會(huì)導(dǎo)致內(nèi)存泄漏,也不會(huì)讓程序的執(zhí)行結(jié)果出乎意料,但 NodeJS 并不是存粹的 JS。NodeJS 里大量的 API 內(nèi)部是用 C/C++ 實(shí)現(xiàn)的,因此 NodeJS 程序的運(yùn)行過程中,代碼執(zhí)行路徑穿梭于 JS 引擎內(nèi)部和外部,而 JS 的異常拋出機(jī)制可能會(huì)打斷正常的代碼執(zhí)行流程,導(dǎo)致 C/C++ 部分的代碼表現(xiàn)異常,進(jìn)而導(dǎo)致內(nèi)存泄漏等問題。

因此,使用 uncaughtException 或 domain 捕獲異常,代碼執(zhí)行路徑里涉及到了 C/C++ 部分的代碼時(shí),如果不能確定是否會(huì)導(dǎo)致內(nèi)存泄漏等問題,最好在處理完異常后重啟程序比較妥當(dāng)。而使用 try 語句捕獲異常時(shí)一般捕獲到的都是 JS 本身的異常,不用擔(dān)心上訴問題。



注:相關(guān)教程知識(shí)閱讀請(qǐng)移步到JavaScript/Ajax教程頻道。
發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 黄梅县| 临江市| 洛浦县| 辰溪县| 五华县| 分宜县| 天柱县| 盐山县| 兰州市| 孝昌县| 太仆寺旗| 宁明县| 南部县| 万年县| 离岛区| 乌拉特中旗| 肃南| 师宗县| 江源县| 临海市| 普定县| 台湾省| 江安县| 泰安市| 靖西县| 嘉黎县| 大英县| 沁阳市| 鄂伦春自治旗| 航空| 鲁甸县| 马鞍山市| 绥滨县| 延川县| 丰县| 嵊泗县| 科尔| 镇坪县| 郸城县| 肥西县| 突泉县|