JS?try?catch用法舉例之異常處理
一、語(yǔ)法
try語(yǔ)句包含了由一個(gè)或者多個(gè)語(yǔ)句組成的try塊,和至少一個(gè)catch塊或者一個(gè)finally塊的其中一個(gè),或者兩個(gè)兼有,下面是三種形式的try聲明:
try...catch try...finally try...catch...finally
try {
// 需要被執(zhí)行的語(yǔ)句
} catch (err) {
// 如果在try塊里有異常被拋出時(shí)執(zhí)行的語(yǔ)句
} finally {
// 在try語(yǔ)句塊之后執(zhí)行的語(yǔ)句塊。無論是否有異常拋出或捕獲這些語(yǔ)句都將執(zhí)行。
}
上述代碼中,
try塊中的語(yǔ)句首先被執(zhí)行。如果運(yùn)行中發(fā)生了錯(cuò)誤,控制就會(huì)轉(zhuǎn)移到位于catch塊中語(yǔ)句,其中括號(hào)中的err參數(shù)被作為例外變量傳遞。否則,catch塊的語(yǔ)句被跳過不執(zhí)行。無論是發(fā)生錯(cuò)誤還是catch塊中的語(yǔ)句執(zhí)行完畢,或者沒有發(fā)生任何錯(cuò)誤try塊中的語(yǔ)句執(zhí)行完畢,最后將執(zhí)行finally塊中的語(yǔ)句。
二、舉例
2.1. 無錯(cuò)誤情況
try {
alert('開始執(zhí)行 try1'); // (1)
alert('開始執(zhí)行 try2'); // (2)
} catch (err) {
alert('catch 被忽略,因?yàn)闆]有 error,不會(huì)被執(zhí)行'); // (3)
}
上面代碼會(huì)彈出‘開始執(zhí)行
try1’和開始執(zhí)行try2,但是不會(huì)彈出‘catch被忽略,因?yàn)闆]有error
2. 2. 有錯(cuò)誤情況
try {
alert('開始執(zhí)行 try1'); // (1)
throw "此處拋出錯(cuò)誤"; //try塊里到此結(jié)束,不會(huì)在執(zhí)行下邊的try2
alert('try2,不會(huì)被執(zhí)行'); // (2)
} catch (err) {
alert('出現(xiàn)了 error,此句被執(zhí)行'); // (3)
}
上面代碼會(huì)彈出‘開始執(zhí)行
try1’和‘出現(xiàn)了error,此句被執(zhí)行’
2.3. try…catch 僅對(duì)可以運(yùn)行的代碼有效
要使得
try...catch能工作,代碼必須是可執(zhí)行的。換句話說,它必須是有效的JavaScript代碼。
如果代碼包含語(yǔ)法錯(cuò)誤,那么try..catch將無法正常工作,例如下邊代碼出現(xiàn)了一行逗號(hào):
try {
,,,,,,//js引擎無法理解這段代碼,它是無效的
throw "此處不會(huì)拋出錯(cuò)誤"; //不會(huì)執(zhí)行
alert('try2,此句不會(huì)被執(zhí)行'); // (2)
} catch (err) {
alert('此句不會(huì)被執(zhí)行'); // (3)
}
2.4. try catch 嵌套
try {
try {
throw new Error("try1");
}
finally {
console.log("finally");
}
}
catch (err) {
console.error("outer", err.message);
}
會(huì)輸出
// finally
// outer try1
try {
try {
throw new Error("try1");
}
catch (err) {
console.error("inner", err.message);
}
finally {
console.log("finally");
}
}
catch (err) {
console.error("outer", err.message);
}
會(huì)輸出
// inner try1
// finally
try {
try {
throw new Error("try1");
}
catch (err) {
console.error("inner", err.message);
// return 如果我在此處寫個(gè)return會(huì)是什么結(jié)果? (下邊的報(bào)錯(cuò)信息肯定不會(huì)拋出了,當(dāng)然return必須是在函數(shù)里)
throw new Error("try2"); //試想一下,如何把此句代碼放在下邊的finally里會(huì)怎么輸出?(結(jié)果一樣)
}
finally {
console.log("finally");
}
}
catch (err) {
console.error("outer", err.message);
}
會(huì)輸出
// inner try1
// finally
// outer try2
try {
try {
throw new Error("try1");
}
catch (err) {
console.error("inner", err.message);
throw new Error("try2");
}
finally {
console.log("finally");
return //終止了錯(cuò)誤的拋出
}
}
catch (err) {
console.error("outer", err.message); //不會(huì)被執(zhí)行
}
會(huì)輸出
// inner try1
// finally
2.5. try…catch 無法捕獲異步任務(wù)異常
2.5.1. 異步任務(wù)->宏任務(wù)setTimeout
try {
setTimeout(()=>{
throw new Error('錯(cuò)誤信息')
}, 1000);
} catch (err) {
alert( "alert不會(huì)執(zhí)行" );
}
因?yàn)?nbsp;
try...catch包裹了計(jì)劃要執(zhí)行的函數(shù),該函數(shù)有延遲,這時(shí)js引擎已經(jīng)離開了try...catch結(jié)構(gòu),也就是上下文環(huán)境已經(jīng)改變,所以無法捕獲異步任務(wù)里的錯(cuò)誤。為了捕獲到計(jì)劃的函數(shù)中的異常,那么try...catch必須在這個(gè)setTimeout函數(shù)內(nèi),如下
setTimeout(() => {
try {
throw new Error("錯(cuò)誤信息");
} catch (err) {
alert("alert會(huì)執(zhí)行");
}
}, 1000);
2.5.2. 異步任務(wù)->微任務(wù)promise
try...catch也不會(huì)捕獲微任務(wù)的異常,道理同2.5.1這2段代碼try,catch都不會(huì)捕獲到錯(cuò)誤信息,因?yàn)?code>promise內(nèi)部的錯(cuò)誤不會(huì)冒泡出來,而是被promise吃掉了,只有通過promise.catch才可以捕獲promise的錯(cuò)誤情況。
function test() {
Promise.resolve("1").then(() => {
throw new Error("錯(cuò)誤信息");
});
}
try {
test();
} catch (err) {
alert("不工作");
}
try {
Promise.reject("錯(cuò)誤信息");
} catch (e) {
console.log(e.message);
}
2.6. try…catch async/await 的異常捕獲
async/await可以讓異步代碼同步執(zhí)行,所以可以進(jìn)行異常捕獲
// 例子1
async function test() {
throw new Error("錯(cuò)誤信息");
return 0;
}
test()
.then()
.catch((e) => console.log(e.message)); // 錯(cuò)誤信息(可以捕獲到錯(cuò)誤信息)
// 例子2
function promiseFn() {
return new Promise((resolve, reject) => {
throw new Error("錯(cuò)誤信息");
});
}
async function test() {
try {
const res = await promiseFn();
} catch (e) {
console.log(e.message); //錯(cuò)誤信息(可以捕獲到錯(cuò)誤信息)
}
}
test();
// 例子3
function promiseFn() {
return Promise.reject("錯(cuò)誤信息");
}
async function test() {
try {
const res = await promiseFn();
} catch (e) {
console.log(e); //錯(cuò)誤信息(可以捕獲到錯(cuò)誤信息)
}
}
test();
2.7. 全局監(jiān)聽拋出的錯(cuò)誤
- 以上所有異常,僅通過
try catch、then捕獲同步、異步錯(cuò)誤。 - 這些是局部錯(cuò)誤捕獲手段,當(dāng)我們無法保證所有代碼都處理了異常時(shí),可以進(jìn)行全局異常監(jiān)控
一般有兩種方法:
- window.addEventListener(‘error’)
- window.addEventListener(‘unhandledrejection’)
- error 可以監(jiān)聽所有同步、異步的運(yùn)行時(shí)錯(cuò)誤,但無法監(jiān)聽語(yǔ)法、接口、資源加載錯(cuò)誤。
- 而 unhandledrejection 可以監(jiān)聽到 Promise 中拋出的,未被 .catch 捕獲的錯(cuò)誤。
window.addEventListener("error", function (e) {
var error = e.error;
console.log(error); //錯(cuò)誤信息
});
document.querySelector("button").addEventListener("click", () => {
throw new Error("錯(cuò)誤信息");
});總結(jié)
到此這篇關(guān)于JS try catch用法舉例之異常處理的文章就介紹到這了,更多相關(guān)JS try catch異常處理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
JS 動(dòng)態(tài)判斷PC和手機(jī)瀏覽器實(shí)現(xiàn)代碼
這篇文章主要介紹了JS 動(dòng)態(tài)判斷PC和手機(jī)瀏覽器實(shí)現(xiàn)代碼的相關(guān)資料,需要的朋友可以參考下2016-09-09
利用TypeScript從字符串字面量類型提取參數(shù)類型
這篇文章主要介紹了利用TypeScript從字符串字面量類型提取參數(shù)類型,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-09-09
boostrap模態(tài)框二次彈出清空原有內(nèi)容的方法
今天小編就為大家分享一篇boostrap模態(tài)框二次彈出清空原有內(nèi)容的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-08-08
javascript 緩沖運(yùn)動(dòng)框架的實(shí)現(xiàn)
這篇文章主要介紹了javascript 緩沖運(yùn)動(dòng)框架的實(shí)現(xiàn)的相關(guān)資料,希望通過本能幫助到大家,實(shí)現(xiàn)這樣類似的功能,需要的朋友可以參考下2017-09-09
JS及JQuery對(duì)Html內(nèi)容編碼,Html轉(zhuǎn)義
本文主要介紹了JS及JQuery對(duì)Html內(nèi)容編碼,Html轉(zhuǎn)義的方法。具有很好的參考價(jià)值,下面跟著小編一起來看下吧2017-02-02
javascript性能優(yōu)化之事件委托實(shí)例詳解
這篇文章主要介紹了javascript性能優(yōu)化之事件委托用法,結(jié)合實(shí)例形式對(duì)比分析了JavaScript中事件委托的具體用法與優(yōu)點(diǎn),需要的朋友可以參考下2015-12-12
TypeScript?mixin提升代碼復(fù)用性的方法和原理
在前端開發(fā)中,我們經(jīng)常需要在不同的組件或類之間共享功能代碼,Mixin提供了一種非常靈活的方式,可以讓我們?cè)诓黄茐睦^承關(guān)系的前提下,將功能代碼復(fù)用到多個(gè)對(duì)象中,文章通過代碼示例介紹mixin提升代碼復(fù)用性的方法和好處,需要的朋友可以參考下2023-06-06
Uniapp?實(shí)現(xiàn)全民分銷功能原理解析
這篇文章主要介紹了Uniapp?實(shí)現(xiàn)全民分銷功能,本篇文章主要介紹全民分銷功能實(shí)現(xiàn)原理,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-06-06

