摘要:我們使用單元測試來驗證一下我們使用了配合做單元測試。我們編寫相應的單元測試你會發現,如果出現異常,只是簡單的返回。但是在上面異常拋出的時候,解釋器已經不在中了,因此無法被捕獲。
譯者按: 錯誤是無法避免的,妥善處理它才是最重要的!
原文: A Guide to Proper Error Handling in JavaScript Related Topics:
譯者: Fundebug
為了保證可讀性,本文采用意譯而非直譯。另外,本文版權歸原作者所有,翻譯僅用于學習。
如果你相信墨菲定律的話,任何事情如果會出問題,那么就一定會出問題。對于代碼,即使我們有100%的自信沒有問題,依然有可能出問題。在這篇文章,我們來研究如何處理JavaScript的錯誤。我會先介紹壞的處理方式、好的處理方式,最終介紹異步代碼和Ajax。
個人感覺,事件驅動的編程設計使得JavaScript語言非常的豐富靈活。我們設想瀏覽器就是事件驅動機器,錯誤同樣由它的驅動產生。當一個錯誤觸發,導致某個事件被拋出。從理論上說,錯誤在JavaScript中就是事件。
如果你對此感到陌生,那么暫且不管它。在這篇文章中,我主要關注瀏覽器端的JavaScript。
這篇文章基于JavaScript中的錯誤處理部分的概念。如果你還不熟悉,我建議你先閱讀一下。
Demo演示我們使用的Demo可以在GitHub下載,程序運行起來會呈現如下頁面:
誤,拋出TypeError。下面是該模塊的定義:
// scripts/error.js function error() { var foo = {}; return foo.bar(); }
在error()中定義了一個空對象foo,因此調用foo.bar()會因為未被定義而報錯。我們使用單元測試來驗證一下:
// tests/scripts/errorTest.js it("throws a TypeError", function () { should.throws(error, TypeError); });
我們使用了Mocha配合Should.js做單元測試。
當你克隆了代碼庫并安裝了依賴包以后,你可以使用npm t來執行測試。當然,你也可以執行某個測試文件,比如:./node_modules/mocha/bin/mocha tests/scripts/errorTest.js
相信我,像JavaScript這樣的動態語言來說,不管誰都很容易遇到這樣的錯誤。
壞的處理方式我已經將按鈕對應的處理事件函數抽象得簡單一點,如下所示:
// scripts/badHandler.js function badHandler(fn) { try { return fn(); } catch (e) { } return null; }
badHandler接收一個fn作為回調函數,該回調函數在badHandler中被調用。我們編寫相應的單元測試:
// tests/scripts/badHandlerTest.js it("returns a value without errors", function() { var fn = function() { return 1; }; var result = badHandler(fn); result.should.equal(1); }); it("returns a null with errors", function() { var fn = function() { throw new Error("random error"); }; var result = badHandler(fn); should(result).equal(null); });
你會發現,如果出現異常,badHandler只是簡單的返回null。如果配合完整的代碼,你會發現問題所在:
// scripts/badHandlerDom.js (function (handler, bomb) { var badButton = document.getElementById("bad"); if (badButton) { badButton.addEventListener("click", function () { handler(bomb); console.log("Imagine, getting promoted for hiding mistakes"); }); } }(badHandler, error));
如果出錯的時候將其try-catch,然后僅僅返回null,我根本找不到哪里出錯了。這種安靜失敗(fail-silent)策略可能導致UI紊亂也可能導致數據錯亂,并且在Debug的時候可能花了幾個小時卻忽略了try-catch里面的代碼才是致禍根源。如果代碼復雜到有多層次的調用,簡直不可能找到哪里出了錯。因此,我們不建議使用安靜失敗策略,我們需要更加優雅的方式。
不壞但很爛的方式// scripts/uglyHandler.js function uglyHandler(fn) { try { return fn(); } catch (e) { throw new Error("a new error"); } }
它處理錯誤的方式是抓到錯誤e,然后拋出一個新的錯誤。這樣做的確優于之前安靜失敗的策略。如果出了錯,我可以一層層找回去,直到找到原本拋出的錯誤e。簡單的拋出一個Error("a new error")信息量比較有限,不精確,我們來自定義錯誤對象,傳出更多信息:
// scripts/specifiedError.js // Create a custom error var SpecifiedError = function SpecifiedError(message) { this.name = "SpecifiedError"; this.message = message || ""; this.stack = (new Error()).stack; }; SpecifiedError.prototype = new Error(); SpecifiedError.prototype.constructor = SpecifiedError; // scripts/uglyHandlerImproved.js function uglyHandlerImproved(fn) { try { return fn(); } catch (e) { throw new SpecifiedError(e.message); } } // tests/scripts/uglyHandlerImprovedTest.js it("returns a specified error with errors", function () { var fn = function () { throw new TypeError("type error"); }; should.throws(function () { uglyHandlerImproved(fn); }, SpecifiedError); });
現在,這個自定義的錯誤對象包含了原本錯誤的信息,因此變得更加有用。但是因為再度拋出來,依然是未處理的錯誤。
截獲異常一個思路是對所有的函數用try...catch包圍起來:
function main(bomb) { try { bomb(); } catch (e) { // Handle all the error things } }
但是,這樣的代碼將會變得非常臃腫、不可讀,而且效率低下。是否還記得?在本文開始我們有提到在JavaScript中異常不過也是一個事件而已,幸運的是,有一個全局的異常事件處理方法(onerror)。
// scripts/errorHandlerDom.js window.addEventListener("error", function (e) { var error = e.error; console.log(error); });獲取堆棧信息
你可以將錯誤信息發送到服務器:
// scripts/errorAjaxHandlerDom.js window.addEventListener("error", function (e) { var stack = e.error.stack; var message = e.error.toString(); if (stack) { message += " " + stack; } var xhr = new XMLHttpRequest(); xhr.open("POST", "/log", true); // Fire an Ajax request with error details xhr.send(message); });
為了獲取更詳細的報錯信息,并且省去處理數據的麻煩,你也可以使用fundebug的JavaScript監控插件三分鐘快速接入bug監控服務。
下面是服務器接收到的報錯消息:
如果你的腳本是放在另一個域名下,如果你不開啟CORS,除了Script error.,你將看不到任何有用的報錯信息。如果想知道具體解法,請參考:Script error.全面解析。
異步錯誤處理由于setTimeout異步執行,下面的代碼異常將不會被try...catch捕獲:
// scripts/asyncHandler.js function asyncHandler(fn) { try { // This rips the potential bomb from the current context setTimeout(function () { fn(); }, 1); } catch (e) { } }
try...catch語句只會捕獲當前執行環境下的異常。但是在上面異常拋出的時候,JavaScript解釋器已經不在try...catch中了,因此無法被捕獲。所有的Ajax請求也是這樣。
我們可以稍微改進一下,將try...catch寫到異步函數的回調中:
setTimeout(function () { try { fn(); } catch (e) { // Handle this async error } }, 1);
不過,這樣的套路會導致項目中充滿了try...catch,代碼非常不簡潔。并且,執行JavaScript的V8引擎不鼓勵在函數中使用try...catch。好在,我們不需要這么做,全局的錯誤處理onerror會捕獲這些錯誤。
結論我的建議是不要隱藏錯誤,勇敢地拋出來。沒有人會因為代碼出現bug導致程序崩潰而羞恥,我們可以讓程序中斷,讓用戶重來。錯誤是無法避免的,如何去處理它才是最重要的。
版權聲明:
轉載時請注明作者Fundebug以及本文地址:
https://blog.fundebug.com/201...
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/90223.html
摘要:如何良好的在代碼中設計異常機制本身設計的出發點是極好的,通過編譯器的強制捕獲,可以明確提醒調用者處理異常情況。但使用此種異常后,該會像病毒一樣,得不到處理后會污染大量代碼,同時也可能因為調用者的不當處理,會失去異常信息。 1、異常是什么? 父類為Throwable,有Error和Exception兩個子類 Error為系統級別的異常(錯誤) Exception下有眾多子類,常見的有Ru...
摘要:對我們來說最大的便利就是利用日志進行錯誤發現和排查的效率變高了。官方也提倡正確設置接收的日志的同時,用戶也能繼續舊的日志備份。 在各種系統和應用里,無論你的代碼再完美也還是會拋異常,出錯誤。今天的主角是當今比較流行的異常記錄框架 - Sentry,來了解一下。 關于日志管理 應用越做越復雜,輸出日志五花八門,有print的,有寫stdout的,有寫stderr的, 有寫logging的...
摘要:但是往往越簡單的東西越容易讓我們忽視,從而導致一些不該有的發生,作為一名嚴謹的程序員,怎么能讓這種事情發生呢所以下面我們就來了解一下關于日志的那些正確使用姿勢。級別表示出現了嚴重錯誤,程序將會中斷執行。 前言 關于日志,在大家的印象中都是比較簡單的,只須引入了相關依賴包,剩下的事情就是在項目中盡情的打印我們需要的信息了。但是往往越簡單的東西越容易讓我們忽視,從而導致一些不該有的bug發...
摘要:本文已收錄修煉內功躍遷之路的為解決空的問題帶來了很多新思路,查看源碼,實現非常簡單,邏輯也并不復雜。 本文已收錄【修煉內功】躍遷之路 showImg(https://segmentfault.com/img/bVbrCvp?w=852&h=480); Java8的Optional為解決空的問題帶來了很多新思路,查看Optional源碼,實現非常簡單,邏輯也并不復雜。Stuart Ma...
閱讀 3778·2021-09-02 09:53
閱讀 2755·2021-07-30 14:57
閱讀 3499·2019-08-30 13:09
閱讀 1202·2019-08-29 13:25
閱讀 815·2019-08-29 12:28
閱讀 1461·2019-08-29 12:26
閱讀 1136·2019-08-28 17:58
閱讀 3310·2019-08-26 13:28