Express框架進(jìn)階3---錯(cuò)誤處理

Error Handling(https://www.expressjs.com.cn/guide/error-handling.html)

本文的【錯(cuò)誤處理】指Express如何捕獲和處理(同步和異步的)錯(cuò)誤。Express有一個(gè)默認(rèn)的錯(cuò)誤處理handler捣卤,所以一開始不是必須寫自己的錯(cuò)誤處理handler爱只。

捕獲錯(cuò)誤
Catching Errors

保證Express會(huì)捕獲運(yùn)行路由handlers和中間件時(shí)發(fā)生的所有錯(cuò)誤非常重要蝙砌。
在路由handlers和中間件的同步代碼中發(fā)生的錯(cuò)誤不需要做額外的工作(Errors that occur in synchronous code inside route handlers and middleware require no extra work),Express會(huì)捕獲和處理适肠。
但路由handlers和中間件中調(diào)用的異步函數(shù)拋出的錯(cuò)誤耿币,我們就需要將異常捕獲,使用next()函數(shù) 將錯(cuò)誤傳入(next(error))容贝, 使Express能夠捕獲和處理這個(gè)錯(cuò)誤。
tips:next函數(shù)除了傳入字符串'route'之景,其他參數(shù)都會(huì)被認(rèn)為是錯(cuò)誤嗤疯,會(huì)跳過余下所有非錯(cuò)誤處理的路由和中間件函數(shù)

// 
app.get('/', [
  function(req, res, next) {
    fs.writeFile('/file-does-not-exit', data, function(err, data){
      if(err) {
        next(err);
      } else {
        res.send(data);
      }
    });
  }
])
// 直接將next作為callback函數(shù),
//有錯(cuò)誤時(shí)next有參數(shù) 攜帶錯(cuò)誤參數(shù)被Express error handler捕獲并處理闺兢,
// 無錯(cuò)誤時(shí) next無參數(shù) , 執(zhí)行下一個(gè)handler
app.get('/', [
  function(req, res, next) {
    fs.writeFile('/file-does-not-exit', data, function(err, data){
      if(err) {
        next(err);
      } else {
        res.send(data);
      }
    });
  }
])
// 需捕獲異常
app.get('/', function (req, res, next) {
  setTimeout(function () {
    try {
      throw new Error('BROKEN')
    } catch (err) {
      next(err)
    }
  }, 100)
})
// 使用promise 代替 try catch
app.get('/', function (req, res, next) {
  Promise.resolve().then(function(){
    throw new Error('BROKEN')
  }).catch(next)
})

默認(rèn)的錯(cuò)誤處理函數(shù)
The default error handler

沒有自定義錯(cuò)誤處理函數(shù),用next(error)傳入錯(cuò)誤的時(shí)候戏罢,express默認(rèn)的錯(cuò)誤處理函數(shù)進(jìn)行處理屋谭,會(huì)同時(shí)輸出調(diào)用棧信息(NODE_ENV的值為production時(shí)不會(huì)打印調(diào)用棧信息)
需要注意的一點(diǎn)是,如果已經(jīng)開始寫response龟糕,調(diào)用了next(error), Express的默認(rèn)錯(cuò)誤處理函數(shù)會(huì)關(guān)閉連接桐磁,此次請(qǐng)求會(huì)失敗。
所以在自定義的error handler中讲岁,當(dāng)header已經(jīng)被發(fā)送給client端我擂,發(fā)生錯(cuò)誤的話需要將錯(cuò)誤傳給默認(rèn)的錯(cuò)誤處理函數(shù)(???)

如果你在你的代碼調(diào)用next()出現(xiàn)錯(cuò)誤多次,則會(huì)觸發(fā)默認(rèn)錯(cuò)誤處理程序缓艳,即使自定義錯(cuò)誤處理中間件已就緒也是如此(???)

function errorHandler (err, req, res, next) {
  if (res.headersSent) {
    return next(err)
  }
  res.status(500)
  res.render('error', { error: err })
}

如何實(shí)現(xiàn)錯(cuò)誤處理函數(shù)
Writing error handlers

定義錯(cuò)誤處理中間件函數(shù)的方式與定義其他中間件函數(shù)一樣校摩,只是錯(cuò)誤處理中間件的參數(shù)是4個(gè) err, req,res ,next

我們需要在最后定義錯(cuò)誤處理中間件函數(shù),在其他的app.use 和路由調(diào)用之后阶淘。

中間件函數(shù)的返回值可以是任意格式衙吩,如html錯(cuò)誤頁面,簡(jiǎn)單的字符串信息 或json字符串
如果在error handler中沒有調(diào)用next函數(shù)溪窒,則需要負(fù)責(zé)返回響應(yīng)坤塞,否則請(qǐng)求會(huì)被掛起冯勉,并且不會(huì)被垃圾回收機(jī)制回收

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市摹芙,隨后出現(xiàn)的幾起案子灼狰,更是在濱河造成了極大的恐慌,老刑警劉巖浮禾,帶你破解...
    沈念sama閱讀 223,002評(píng)論 6 519
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件交胚,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡伐厌,警方通過查閱死者的電腦和手機(jī)承绸,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,357評(píng)論 3 400
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來挣轨,“玉大人军熏,你說我怎么就攤上這事【戆纾” “怎么了荡澎?”我有些...
    開封第一講書人閱讀 169,787評(píng)論 0 365
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)晤锹。 經(jīng)常有香客問我摩幔,道長(zhǎng)耽装,這世上最難降的妖魔是什么娶聘? 我笑而不...
    開封第一講書人閱讀 60,237評(píng)論 1 300
  • 正文 為了忘掉前任衔统,我火速辦了婚禮掂咒,結(jié)果婚禮上墅拭,老公的妹妹穿的比我還像新娘失仁。我一直安慰自己掀序,他們只是感情好爹脾,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,237評(píng)論 6 398
  • 文/花漫 我一把揭開白布舶担。 她就那樣靜靜地躺著坡疼,像睡著了一般。 火紅的嫁衣襯著肌膚如雪衣陶。 梳的紋絲不亂的頭發(fā)上柄瑰,一...
    開封第一講書人閱讀 52,821評(píng)論 1 314
  • 那天,我揣著相機(jī)與錄音剪况,去河邊找鬼教沾。 笑死,一個(gè)胖子當(dāng)著我的面吹牛拯欧,可吹牛的內(nèi)容都是我干的详囤。 我是一名探鬼主播,決...
    沈念sama閱讀 41,236評(píng)論 3 424
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼藏姐!你這毒婦竟也來了隆箩?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 40,196評(píng)論 0 277
  • 序言:老撾萬榮一對(duì)情侶失蹤羔杨,失蹤者是張志新(化名)和其女友劉穎捌臊,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體兜材,經(jīng)...
    沈念sama閱讀 46,716評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡理澎,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,794評(píng)論 3 343
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了曙寡。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片糠爬。...
    茶點(diǎn)故事閱讀 40,928評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖举庶,靈堂內(nèi)的尸體忽然破棺而出执隧,到底是詐尸還是另有隱情,我是刑警寧澤户侥,帶...
    沈念sama閱讀 36,583評(píng)論 5 351
  • 正文 年R本政府宣布镀琉,位于F島的核電站,受9級(jí)特大地震影響蕊唐,放射性物質(zhì)發(fā)生泄漏屋摔。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,264評(píng)論 3 336
  • 文/蒙蒙 一替梨、第九天 我趴在偏房一處隱蔽的房頂上張望钓试。 院中可真熱鬧,春花似錦副瀑、人聲如沸亚侠。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,755評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至箕别,卻和暖如春铜幽,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背串稀。 一陣腳步聲響...
    開封第一講書人閱讀 33,869評(píng)論 1 274
  • 我被黑心中介騙來泰國(guó)打工除抛, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人母截。 一個(gè)月前我還...
    沈念sama閱讀 49,378評(píng)論 3 379
  • 正文 我出身青樓到忽,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子喘漏,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,937評(píng)論 2 361