javascript
被吐嘈的NodeJS的异常处理
?
被吐嘈的NodeJS的異常處理
許多人都有這樣一種映像,NodeJS比較快; 但是因為其是單線程,所以它不穩(wěn)定,有點不安全,不適合處理復(fù)雜業(yè)務(wù); 它比較適合對并發(fā)要求比較高,而且簡單的業(yè)務(wù)場景。?
在Express的作者的TJ Holowaychuk的?告別Node.js一文中列舉了以下罪狀:?
Farewell NodeJS (TJ Holowaychuk)?
??? you may get duplicate callbacks?
??? you may not get a callback at all (lost in limbo)?
??? you may get out-of-band errors?
??? emitters may get multiple “error” events?
??? missing “error” events sends everything to hell?
??? often unsure what requires “error” handlers?
??? “error” handlers are very verbose?
??? callbacks suck?
其實這幾條主要吐嘈了兩點: node.js錯誤處理很扯蛋,node.js的回調(diào)也很扯蛋。?
事實上呢?
事實上NodeJS里程確實有“脆弱”的一面,單線程的某處產(chǎn)生了“未處理的”異常確實會導(dǎo)致整個Node.JS的崩潰退出,來看個例子, 這里有一個node-error.js的文件:?
啟動服務(wù),并在地址欄測試一下發(fā)現(xiàn)?http://127.0.0.1:8080/? 不出所料,node崩潰了?
怎么解決呢?
其實Node.JS發(fā)展到今天,如果連這個問題都解決不了,那估計早就沒人用了。?
使用uncaughtException
我們可以uncaughtException來全局捕獲未捕獲的Error,同時你還可以將此函數(shù)的調(diào)用棧打印出來,捕獲之后可以有效防止node進(jìn)程退出,如:?
這相當(dāng)于在node進(jìn)程內(nèi)部進(jìn)行守護, 但這種方法很多人都是不提倡的,說明你還不能完全掌控Node.JS的異常。?
使用 try/catch
我們還可以在回調(diào)前加try/catch,同樣確保線程的安全。?
這種方案的好處是,可以將錯誤和調(diào)用棧直接輸出到當(dāng)前發(fā)生的網(wǎng)頁上。?
集成到框架中
標(biāo)準(zhǔn)的HTTP響應(yīng)處理會經(jīng)歷一系列的Middleware(HttpModule),最終到達(dá)Handler,如下圖所示:?
?
這 些Middleware和Handler在NodeJS中都有一個特點,他們都是回調(diào)函數(shù),而回調(diào)函數(shù)中是唯一會讓Node在運行時崩潰的地方。根據(jù)這個 特點,我們只需要在框架中集成一處try/catch就可以相對完美地解決異常問題,而且不會影響其它用戶的請求request。?
事實上現(xiàn)在的NodeJS WEB框架幾乎都是這么做的,如?OurJS開源博客所基于的?WebSvr?
就有這么一處異常處理代碼:?
那么不在回調(diào)中產(chǎn)生的錯誤怎么辦?不必?fù)?dān)心,其實這樣的node程序根本就起不起來。?
此外node自帶的?cluster?也有一定的容錯能力,它跟nginx的worker很類似,但消耗資源(內(nèi)存)略大,編程也不是很方便,OurJS并沒有采用此種設(shè)計。?
守護NodeJS進(jìn)程和記錄錯誤日志
現(xiàn) 在已經(jīng)基本上解決了Node.JS因異常而崩潰的問題,不過任何平臺都不是100%可靠的,還有一些錯誤是從Node底層拋出的,有些異常 try/catch和uncaughtException都無法捕獲。之前在運行ourjs的時侯,會偶爾碰到底層拋出的文件流讀取異常,這就是一個底層 libuv的BUG,node.js在0.10.21中進(jìn)行了修復(fù)。?
面對這種情況,我們就應(yīng)該為nodejs應(yīng)用添加守護進(jìn)程,讓NodeJS遭遇異常崩潰以后能馬上復(fù)活。?
另外,還應(yīng)該把這些產(chǎn)生的異常記錄到日志中,并讓異常永遠(yuǎn)不再發(fā)生。?
使用node來守護node
node-forever?提供了守護的功能和LOG日志記錄功能。?
安裝非常容易?
使用也很簡單?
還可以看日志?
?
使用shell啟動腳本守護node
使用node來守護的話資源開銷可能會有點大,而且也會略顯復(fù)雜,OurJS直接在開機啟動腳本來進(jìn)程線程守護。?
如在debian中放置的 ourjs 開機啟動文件:?/etc/init.d/ourjs?
這個文件非常簡單,只有啟動的選項,守護的核心功能是由一個無限循環(huán) while true; 來實現(xiàn)的,為了防止過于密集的錯誤阻塞進(jìn)程,每次錯誤后間隔1秒重啟服務(wù)?
?
錯誤日志記錄也非常簡單,直接將此進(jìn)程控制臺當(dāng)中的錯誤輸出到error.log文件即可: 2>> $WEB_DIR/error.log? 這一行, 2 代表 Error。
?
轉(zhuǎn)自:?OurJS
總結(jié)
以上是生活随笔為你收集整理的被吐嘈的NodeJS的异常处理的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 做梦梦到身上长虫是怎么回事
- 下一篇: javascript中对变量类型的推断