JavaScript進階程式設計 閱讀筆記(二十) js錯誤處理

來源:互聯網
上載者:User

一、錯誤分類

  1、語法錯誤:也稱解析錯誤,發生在傳統語言的編譯時間,在JavaScript中發生在解釋時。這些錯誤是由代碼中的意外字元直接引起的,然後就不能直接編譯/解釋。發生語法錯誤時,就不能繼續執行代碼。在JavaScript中,只有在同一個線程中的代碼會受語法錯誤的影響。在其他線程中的代碼和其他外部參考的檔案中的代碼,如果不依賴於包含錯誤的代碼,則可以繼續執行。
  2、執行階段錯誤:也稱為異常(exception,在編譯期/解譯器後)。此時,問題並不出在代碼的文法上,而是,嘗試完成的一個操作,在某些情況下是非法的。異常隻影響發生的線程,其他JavaScript線程即可繼續正常的執行。

二、錯誤處理

  JavaScript提供了兩種處理錯誤的方式:BOM中的onerror事件處理函數方式和ECMAScript中的try...catch方式。
  1、onerror事件處理函數
  它是第一個用來協助JavaScript處理錯誤的機制。頁面上出現異常時,error事件便在window對象上觸發。例如:

複製代碼 代碼如下:<html>
<head>
<title>onerror Example</title>
<script type="text/javascript">
window.onerror = function() {
alert("發生錯誤!");
}
</script>
</head>
<body onload="nonExistentFunction()">
</body>
</html>

在上述代碼中,在頁面載入時嘗試調用不存在的函數,此時會引發一個異常。彈出“發生錯誤”的錯誤資訊。但是,瀏覽器的錯誤資訊也顯示出來了,如何在瀏覽器上隱藏它呢,只需onerror方法返回一個true即可。

複製代碼 代碼如下:<script type="text/javascript">
window.onerror = function() {
alert(“發生錯誤!”);
return true;
}
</script>

1.1 取出錯誤資訊

  onerror處理函數提供了三種資訊來確定錯誤確切的性質:
  i) 錯誤資訊——對於給定錯誤,瀏覽器會顯示同樣的資訊
  ii) URL——在哪個檔案中發生了錯誤
  iii) 行號——給定URL中發生錯誤的行號。 複製代碼 代碼如下:window.onerror = function(sMessage, sUrl, iLine) {
alert("發生錯誤!\n" + sMessage + "\nURL:" + sUrl + "\nLine Number:" + iLine);
return true;
}

  1.2 映像載入錯誤

  window對象並非唯一支援onerror事件處理函數的對象,它對映像對象也提供支援。當一個映像由於檔案不存在等原因未能成功載入時,error事件便在這個映像上觸發。例如: 複製代碼 代碼如下:<img src="noexist.gif" onerror="alert('載入圖片時發生錯誤')"/>  

上例直接在HTML中分配onerror事件處理函數。因為noexist.gif不存在,所以會彈出警告框提示使用者。當然也可以通過指令碼來分配事件處理函數,在設定映像的src特性前,必須等待頁面完全載入,例如: 複製代碼 代碼如下:<html>
<head>
<title>Image錯誤測試</title>
<script type="text/javascript">
function handleLoad() {
document.images[0].onerror = function() {
alert("載入圖片時發生錯誤!");
};
document.images[0].src = "amigo.jpg";
}
</script>
</head>
<body onload="handleLoad()">
<img/>
<body>
</html>

注意:與window對象的onerror事件處理函數不同,image的onerror事件沒有任何關於額外資訊的參數。

  1.3處理語法錯誤

  onerror事件處理函數不僅可以處理異常,它還能處理語法錯誤,也只有它才能處理。
  首先,事件處理函數必須是頁面中第一個出現的代碼,因為如果語法錯誤出現在設定事件處理函數之前出現,事件處理函數就沒有用了。記住,語法錯誤會完全停止代碼的執行。例如: 複製代碼 代碼如下:<html>
<head>
<title>onError Example</title>
<script type="text/javascript">
alert("Syntax error.";
window.onerror = function (sMessage, sUrl, iLine){
alert("An error occurrred:\n" + sMessage + "\nURL:" + sUrl + "\nLine Number:" + iLine );
return true;
}
</script>
</head>
<body onload="nonExistentFunction()">
</body>
</html>

因為反白的那一行代碼(裡面有錯誤文法)在分配onerror事件處理函數之前就出現了,所以瀏覽器直接報告這個錯誤。在錯誤之後的代碼就不再被解釋(因為這個線程已經退出了),所以 load 事件解發時調用 nonExistentFunction() ,瀏覽器也會報這個錯誤。書中說如果重寫這個頁面,將 onerror 事件處理函數的分配放在語法錯誤之前,那麼會出現兩個警告框:一個顯示語法錯誤,另一個顯示異常。但我測試的結果還是一樣的報兩個錯誤,並不顯示onerror事件中的資訊。

  使用onerror事件處理函數的主要的問題是,它是BOM的一部分,所以,沒有任何標準能控制它的行為。因此,不同的瀏覽器使用這個事件處理錯誤的方式有明顯的不同,eg,在IE中發生error事件時,正常的代碼會繼續執行,所有的變數和資料都保留下來,並可通過onerror事件處理函數訪問。在Mozilla中,正常的代碼執行都會結束,同時所有的錯誤發生之前的變數和資料都被銷毀。

  2、try...catch方式

  ECMPScript第三版,引入了try…catch語句。基本文法如下: 複製代碼 代碼如下:try{
//code
[break;]
} catch ([exception]) {
//code
[break;]
} [finally{
//code
}]

例如: 複製代碼 代碼如下:try {
window.openFile1();
alert("成功調用openFile1方法");
} catch (exception) {
alert("發生異常!");
} finally {
alert("try..catch測試結束!");
}

與Java不同,ECMAScript標準在try…catch語句中只能有一個catch語句,因為JavaScript是弱類型的語言,沒辦法指明catch子句中異常的特定類型。不管錯誤是什麼類型,都由同一個catch語句處理。Mozilla對其進行了擴充,可加多個catch語句,但因為只有 Mozilla 可以使用,因此不推薦使用。

  finally用於包含無論是否有異常發生都要執行的代碼,這對關閉開啟的連結和釋放資源很有用。

  2.1 嵌套 try...catch 語句

  在 try...catch 語句中的 catch 子句中,也會發生錯誤。此時,就可以使用嵌套的 try...catch 語句。樣本: 複製代碼 代碼如下:try {
eval("a ++ b");
} catch(oException) {
alert("發生錯誤!");
try {
var aError = new Array(1000000000000000000000000000000000000000);
} catch(exception) {
alert("在catch子句中發生錯誤!");
}
} finally{
alert("已完成")
}

2.2 Error對象

  發生錯誤時,JavaScript有個Error基類用於拋出。它有兩個特性:
  i)name——表示錯誤類型的字串
  ii)message——實際的錯誤資訊
  Error對象的name對應於它的類,可以是如下值之一:
  EvalError:錯誤發生在eval()函數中;
  RangeError:數字值超出JavaScript可表示的範圍;
  ReferenceError:使用了非法的引用;
  SyntaxError:在eval()函數調用中發生了語法錯誤,其他的愈發錯誤由瀏覽器報告,無法通過try…catch處理;
  TypeError:變數的類型不是預期所需的;
  URIError:在encodeURI或decodeURI函數發生了錯誤。

  2.3 判斷錯誤類型

  儘管每個 try...catch 語句中只能有一個catch子句,但判斷拋出的錯誤類型方法主要有兩種。第一種使用 Error 對象的 name 特性: 複製代碼 代碼如下:try {
eval("a ++ b");
} catch(oException) {
if (oException.name = "SyntaxError") {
alert("發生SyntaxError!");
} else {
alert("發生其他錯誤!");
}
}

第二種使用 instanceof 操作符,並使用不同錯誤的類名: 複製代碼 代碼如下:try {
eval("a ++ b");
} catch(oException) {
if (oException instanceof SyntaxError) {
alert("發生SyntaxError!");
} else {
alert("發生其他錯誤!");
}
}

2.4 拋出異常

  在ECMAScript第三版引入,用於有目的的拋出異常,拋出的錯誤對象可為字串、數字、布爾值或實際的對象,也可以拋出Error對象(其建構函式只有一個函數,即錯誤資訊)。如: 複製代碼 代碼如下:throw new Error("錯誤產生!");  

開發人員拋出的錯誤和由瀏覽器自身拋出的錯誤都在 try...catch 中捕獲。例如: 複製代碼 代碼如下:function addTwoNumber(a, b) {
if (arguments.length < 2) {
throw new Error("需要傳入兩個數字!");
}
}
try {
result = addTwoNumber(90);
} catch(oException) {
if (oException instanceof SyntaxError) {
alert("SyntaxError:" + oException.message);
} else if (oException instanceof Error){
alert(oException.message);
}
}

三、調試技巧

  現在的瀏覽器大多都內建了調試工具,大多數情況下已經夠用了,另外IE下還可以用IETest,FireFox下還可以用FireBug。

作者:Artwl
出處:http://artwl.cnblogs.com

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.