最近發現一個基於Node.js平臺上的Express框架運行的Web網站經常報這樣一個錯誤: 網站的源碼中有專門針對錯誤處理的中間件,經過精簡之後的代碼如下: 乍一看,這裡的status code不太可能為0,因為無論err.code的值為字元串0還是數字0,最終都會被賦值為500。除非err.co ...
最近發現一個基於Node.js平臺上的Express框架運行的Web網站經常報這樣一個錯誤:
RangeError: Invalid status code: 0
網站的源碼中有專門針對錯誤處理的中間件,經過精簡之後的代碼如下:
module.exports = function (err, req, res, next) { var _code = err.code || 500; if (_code < 100 || _code >= 600) { _code = 500; } var _finalErr = {statusCode: -1, message: '500 - 伺服器內部錯誤', code: _code, err: err, error: true}; res.status(_code); if (!res.headersSent) { res.json(_finalErr); } if (err) { next(err); } };
乍一看,這裡的status code不太可能為0,因為無論err.code的值為字元串0還是數字0,最終都會被賦值為500。除非err.code的原始值是一個不能隱式轉換成數字的字元串。為了進行驗證,我們寫瞭如下代碼:
var _err = new Error(); _err.code = "illegal http status code"; throw _err;
啟動WebStorm進入調試模式,果然復現了本文開頭給出的那個錯誤。
那麼問題來了,為什麼給定的http status code是一個字元串,錯誤提示卻顯示這裡的status code是0呢?為了搞清楚其中的原因,我們根據錯誤堆棧一層層查找源碼。該錯誤的最終出處是在Node.js源碼的_http_server.js文件的writeHead函數中,核心部分的代碼如下:
statusCode |= 0; if (statusCode < 100 || statusCode > 999) throw new RangeError(`Invalid status code: ${statusCode}`); if (common._checkInvalidHeaderChar(this.statusMessage)) throw new Error('Invalid character in statusMessage.');
這裡使用了javascript中的按位或運算符:| 。其目的是將所有非數字的statusCode都預設轉換成0。可以參考以下兩篇文章中的描述來理解javascript中的位運算符:
http://www.w3school.com.cn/js/pro_js_operators_bitwise.asp
值得註意的是,當參與計算的值不能隱式轉換成數字時,得到的結果為0,可以參考上面第二篇文章中給出的一些實際例子。
在實際應用中,巧妙使用位運算符,可以非常方便地實現我們想要的效果,例如,判斷一個給定的數值是否為偶數、找出給定數值最接近的偶數、判斷一個字元串是否包含在另一個字元串中等等。