溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Node中Express的錯誤處理中間件怎么定義

發(fā)布時間:2022-04-07 09:55:21 來源:億速云 閱讀:244 作者:iii 欄目:web開發(fā)

這篇文章主要介紹“Node中Express的錯誤處理中間件怎么定義”,在日常操作中,相信很多人在Node中Express的錯誤處理中間件怎么定義問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Node中Express的錯誤處理中間件怎么定義”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!

Node中Express的錯誤處理中間件怎么定義

Express 的錯誤處理中間件可幫助您處理錯誤,而無需重復(fù)同樣的工作。假設(shè)您直接在 Express 路由處理程序中處理錯誤:

app.put('/user/:id', async (req, res) => {
  let user
  try {
    user = await User.findOneAndUpdate({ _id: req.params.id }, req.body)
  } catch (err) {
    return res.status(err.status || 500).json({ message: err.message })
  }
  return res.json({ user })
})

上面的代碼可以正常工作,但是,如果有數(shù)百個接口呢,那么錯誤處理邏輯將變得不可維護(hù),因為它被重復(fù)了數(shù)百次。

定義錯誤處理中間件

Express 根據(jù)中間件函數(shù)采用的參數(shù)數(shù)量分為不同的類型。接受 4 個參數(shù)的中間件函數(shù)被定義為錯誤處理中間件,只有在發(fā)生錯誤時才會被調(diào)用。

const app = require('express')()

app.get('*', function routeHandler() {
  // 此中間件拋出一個錯誤,Express 將直接轉(zhuǎn)到下一個錯誤處理程序
  throw new Error('Oops!')
})

app.get('*', (req, res, next) => {
  // 此中間件不是錯誤處理程序(只有3個參數(shù)),Express 將跳過它,因為之前的中間件中存在錯誤
  console.log('這里不會打印')
})

// 您的函數(shù)必須接受 4 個參數(shù),以便 Express 將其視為錯誤處理中間件。
app.use((err, req, res, next) => {
  res.status(500).json({ message: err.message })
})

Express 會自動為您處理同步錯誤,如上面的 routeHandler() 方法。但是 Express 不處理異步錯誤。如果出現(xiàn)異步錯誤,則需要調(diào)用 next()。

const app = require('express')()

app.get('*', (req, res, next) => {
  // next() 方法告訴 Express 轉(zhuǎn)到鏈中的下一個中間件。
  // Express 不處理異步錯誤,因此您需要通過調(diào)用 next() 來報告錯誤。
  setImmediate(() => {
    next(new Error('Oops'))
  })
})

app.use((err, req, res, next) => {
  res.status(500).json({
    message: err.message
  })
})

請記住,Express 中間件是按順序執(zhí)行的。您應(yīng)該在所有其他中間件之后,最后定義錯誤處理程序。否則,您的錯誤處理程序?qū)⒉粫徽{(diào)用:

async/await 一起使用

Express 無法捕獲 promise 的異常,Express 在 ES6 之前編寫,對于如何處理 async/await 它扔沒有好的解決方案。

例如,下面的服務(wù)器永遠(yuǎn)不會成功發(fā)送 HTTP 響應(yīng),因為 Promise reject 永遠(yuǎn)不會得到處理:

const app = require('express')()

app.get('*', (req, res, next) => {
  // 報告異步錯誤必須通過 next()
  return new Promise((resolve, reject) => {
    setImmediate(() => reject(new Error('woops')))
  }).catch(next)
})

app.use((error, req, res, next) => {
  console.log('will not print')
  res.json({ message: error.message })
})

app.listen(3000)

我們可以封裝或者使用現(xiàn)有的庫來進(jìn)行捕獲。

首先,我們先簡單封裝一個函數(shù),將 async/await 與 Express 錯誤處理中間件聯(lián)系起來。

注意:異步函數(shù)會返回 Promise,因此您需要確保 catch() 所有錯誤并將其傳遞給 next()。

function wrapAsync(fn) {
  return function(req, res, next) {
    fn(req, res, next).catch(next)
  }
}

app.get('*', wrapAsync(async (req, res) => {
  await new Promise(resolve => setTimeout(() => resolve(), 50))
  // Async error!
  throw new Error('woops')
}))

使用第三方庫 express-async-errors,一個簡單的 ES6 async/await 支持 hack:

require('express-async-errors')
app.get('*', async (req, res, next) => {
  await new Promise((resolve) => setTimeout(() => resolve(), 50))
  throw new Error('woops')
})

到此,關(guān)于“Node中Express的錯誤處理中間件怎么定義”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注億速云網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI