《javascript高級程序設計》學習筆記 | 21.2.錯誤處理

關注[前端小謳],原創技術文章

錯誤處理

相關代碼 →

try/catch 語句

  • ES3 新增了try/catch語句,基本語法與 Java 中的 try/catch 一樣
try {// 可能出錯的代碼const a = 3;a = 4;
} catch (error) {// 出錯時執行的代碼console.log("An error happened!"); // An error happened!
}
  • try 塊中有代碼發生錯誤,代碼會立即退出執行并跳到 catch 塊中
  • 所有瀏覽器都支持錯誤對象的messagename屬性
try {const a = 3;a = 4;
} catch (error) {console.log(error);/* TypeError: Assignment to constant variable.at Object.<anonymous> (c:\Users\43577\Desktop\工作\my_project\my_demos\javascript高級程序設計(第四版)\第21章 錯誤處理與調試\21.2.錯誤處理.js:13:5)at Module._compile (internal/modules/cjs/loader.js:1085:14)at Object.Module._extensions..js (internal/modules/cjs/loader.js:1114:10)at Module.load (internal/modules/cjs/loader.js:950:32)at Function.Module._load (internal/modules/cjs/loader.js:790:12)at Function.executeUserEntryPoint [as runMain] (internal/modules/run_main.js:75:12)at internal/main/run_main_module.js:17:47*/console.log(error.name); // TypeError(類型錯誤)console.log(error.message); // Assignment to constant variable.(常量被賦值)
}

finally 子句

  • try/catch 中可選的 finally 子句始終運行,二者均無法阻止finally 塊執行
    • try 中代碼運行完,會執行 finally 的代碼
    • 出錯并執行 catch 中代碼,仍會執行 finally 的代碼
// finally 子句
try {console.log(1); // 1,執行
} catch (error) {console.log(2); // 不執行
} finally {console.log(3); // 3,執行
}try {const a = 3;a = 4;
} catch (error) {console.log(2); // 2,try出錯執行catch
} finally {console.log(3); // 3,仍執行
}
  • 代碼中包含 finally, try 或 catch 中的 return 會被忽略
console.log((function testFinally() {try {console.log("try"); // try,非return語句不受影響return 1;} catch (error) {return 2;} finally {console.log("finally"); // finallyreturn 3;}})()
); // 3,包含finally語句,try或catch中的return會被忽略

錯誤類型

  • Error,基類型
  • InternalError,底層引擎異常時,如遞歸過多導致的棧溢出
  • EvalError,使用 eval()異常時,但瀏覽器不會總拋出 EvalError
  • RangeError,數值越界時
  • ReferenceError,找不到對象時
  • SyntaxError,給 eval()傳入的字符串包含語法錯誤時
  • TypeError,最常見
    • 變量不是預期類型時
    • 訪問不存在的方法時
new Array(-1); // RangeError: Invalid array length
let obj = x; // ReferenceError: x is not defined
eval("1++2"); // SyntaxError: Invalid left-hand side expression in postfix operation
console.log("a" in "abc"); // TypeError: Cannot use 'in' operator to search for 'a' in abc
Function.prototype.toString().call("name"); // TypeError: Function.prototype.toString(...).call is not a function
  • 可以使用instanceof操作符在 catch 塊中確定錯誤類型
try {const a = 3;a = 4;
} catch (error) {if (error instanceof TypeError) {console.log("TypeError!");}
} // TypeError!
try {new Array(-1);
} catch (error) {if (error instanceof RangeError) {console.log("RangeError!");}
} // RangeError!

try/catch 的用法

  • 瀏覽器認為 try/catch 中發生的錯誤已被處理,不會再報錯
  • try/catch 最好使用在開發者無法控制有可能出現錯誤上(如不便修改代碼的第三方 js 庫,最好使用 try/catch 把函數調用包起來)

拋出錯誤

  • throw 操作符可在任何時候拋出自定義錯誤

    • throw 操作符必須有值,類型不限
    // throw 12345; // Uncaught 12345,后續代碼停止
    // throw "Hello world"; // Uncaught Hello world,后續代碼停止
    // throw true; // Uncaught true,后續代碼停止
    // throw { name: "JS" }; // Uncaught {name: 'JS'},后續代碼停止
    
    • 使用 throw 時代碼立即停止,try/catch 語句中捕獲了拋出的值時除外
    try {throw 123;
    } catch (error) {console.log(123);
    } // 123
    console.log(5); // 5,throw被try/catch捕獲,后續代碼照常
    
  • 可通過內置錯誤類型模擬瀏覽器錯誤

// throw new SyntaxError; //  Uncaught SyntaxError
// throw new InternalError; //  Uncaught InternalError
// throw new TypeError; //  Uncaught TypeError
// throw new RangeError; //  Uncaught RangeError
// throw new EvalError; //  Uncaught EvalError
// throw new URIError; //  Uncaught URIError
// throw new RefenceError; //  Uncaught RefenceError
  • 可通過繼承 Error創建自定義錯誤類型,創建時需提供 name 和 message 屬性
class CustomError extends Error {constructor(message) {super(message); // super調用父類構造函數,手動給父類傳參,并將返回值賦給子類中的thisthis.name = "CustomError";this.message = message;}
}
// throw new CustomError("My message"); // CustomError: My message

何時拋出錯誤

  • 已知函數無法正確執行時,瀏覽器會自動拋出錯誤
  • 復雜的程序很難找到錯誤原因,適當創建自定義錯誤可有效提高代碼的可維護性
  • 應仔細評估每個函數,尤其可能導致失敗的情形
function process(values) {if (!(values instanceof Array)) {throw new Error("process(): Argument must be an Array.");}values.sort(); // 如果values不是數組,則瀏覽器會報錯。因此在此句之前判斷參數類型且用自定義錯誤,可有效提高代碼可維護性for (let value of values) {if (value > 100) {return value;}}return -1;
}
// process(1); // Error: process(): Argument must be an Array.
// process(1); // TypeError: values.sort is not a function(如果沒有throw代碼段的結果)

拋出錯誤與 try/catch

  • 捕獲錯誤的目的是阻止瀏覽器以其默認方式響應
  • 拋出錯誤的目的是提供有關其發生原因的說明
  • 應該在明確接下來做什么時捕獲錯誤

error 事件

  • 沒有被 try/catch 捕獲的錯誤會在瀏覽器 window 對象上觸發 error 事件
    • onerror 事件處理程序中,任何瀏覽器都不傳入 event 對象
    • 傳入 3 個參數:錯誤消息、發生錯誤的 URL、發生錯誤的行號
    • 任何錯誤發生都會觸發 error 事件,并執行事件的處理程序,瀏覽器默認行為會生效
    • 可以返回 false 來阻止瀏覽器默認報告錯誤的行為
window.onerror = (message, url, line) => {console.log(message);return false; // 阻止瀏覽器默認報告錯誤
};
  • 圖片中 src 屬性的 url 沒有返回可識別的圖片格式,也會觸發 error 事件
const image = new Image();
image.addEventListener("load", (event) => {console.log("Image loaded!");
});
image.addEventListener("error", (event) => {console.log("Image not loaded!");
});
image.src = "a.jpg"; // Image not loaded!

識別錯誤

類型轉換錯誤

  • 主要原因是使用了會自動改變某個值的數據類型的草錯付或語言構造

    • 比較過程中,應使用嚴格相等嚴格不等避免錯誤
    console.log(5 == "5"); // true
    console.log(5 === "5"); // false,數據類型不同
    console.log(1 == true); // true
    console.log(1 === true); // false,數據類型不同
    
    • 在 if、for、while 等流程控制語句中,應堅持使用布爾值作為條件避免錯誤
    function concat(str1, str2, str3) {let result = str1 + str2;if (str3) {result += str3;}return result;
    }
    console.log(concat("1", "2", "0")); // '120'
    console.log(concat("1", "2")); // '12',str3是undifined,轉化為false
    console.log(concat("1", "2", 0)); // '12',str3是數值0,轉化為false,與預期不符function concat(str1, str2, str3) {let result = str1 + str2;if (str3 !== undefined) {result += str3;}return result;
    }
    console.log(concat("1", "2", "03")); // '120'
    console.log(concat("1", "2")); // '12',str3 是 undifined,轉化為 false
    console.log(concat("1", "2", 0)); // '120',達到預期
    

數據類型錯誤

  • JS 是松散類型,其變量函數參數不能保證數據類型

    • 原始類型的值,使用typeof檢測
    function getQueryString(url) {const pos = url.indexOf("?"); // indexOf是字符串才有的方法if (pos > 1) {console.log(url.substring(pos + 1)); // substring是字符串才有的方法return;}console.log("not has ?");
    }
    // getQueryString(123); // TypeError: url.indexOf is not a functionfunction getQueryString2(url) {if (typeof url === "string") {// 確保不會因為參數是非字符串值而報錯const pos = url.indexOf("?");if (pos > 1) {console.log(url.substring(pos + 1));return;}console.log("not has ?");}
    }
    getQueryString2(123); // 不打印
    getQueryString2("123"); // 'not has ?'
    getQueryString2("https://www.baidu.com?keyWord=error"); // 'keyWord=error'
    
    • 對象值,使用instanceof檢測
    function reverseSort(values) {if (values) {// 不可取,values為true的情況很多values.sort();values.reverse();}
    }
    // reverseSort(1); // TypeError: values.sort is not a functionfunction reverseSort2(values) {if (values !== null) {// 不可取,values不為null的情況很多values.sort();values.reverse();}
    }
    // reverseSort2(1); // TypeError: values.sort is not a functionfunction reverseSort3(values) {if (typeof values.sort === "function") {// 不可取,假如values有sort()方法但不是數組則會報錯values.sort();values.reverse();}
    }
    // reverseSort3({
    //   sort: () => {
    //     console.log("3");
    //   },
    // }); // 先values.sort()打印3,后報錯TypeError: values.reverse is not a functionfunction reverseSort4(values) {if (values instanceof Array) {// 可取,確保values是Array的實例values.sort();values.reverse();}
    }
    let val1 = 1;
    let val2 = [1, 2];
    reverseSort4(val1);
    reverseSort4(val2);
    console.log(val1); // 1
    console.log(val2); // [2,1]
    

通信錯誤

  • 對于 url 的查詢字符串,都要通過encodeURIComponent(),以確保編碼合適
let url = "https://www.baidu.com?keyWord=https://www.taobao.com"; // url格式不正確
function addQueryStringArg(url, name, value) {if (url.indexOf("?") === -1) {url += "?";} else {url += "&";}url += `${encodeURIComponent(name)}=${encodeURIComponent(value)}`;return url;
}
let url2 = addQueryStringArg("https://www.baidu.com","keyWord","https://www.taobao.com"
);
console.log(url2); // https://www.baidu.com?keyWord=https%3A%2F%2Fwww.taobao.com,與服務器通信的正確url格式

區分重大與非重大錯誤

  • 非重大錯誤
    • 不影響用戶主要任務
    • 只影響頁面中某個部分
    • 可恢復
    • 重復操作可能成功
  • 重大錯誤
    • 程序無法繼續運行
    • 嚴重影響用戶的主要目標
    • 會導致其他錯誤
  • 程序某個部分的錯誤,不應該影響其他部分
  • 模塊初始化時,可在 for 循環中加入 try/catch 語句,避免某一模塊初始化時發生錯誤影響其他模塊
let mods = [{name: "mod1",init: () => {const a = 1;a = 2;console.log("mod1 init");}, // mod1的init方法里有錯誤},{name: "mod2",init: () => {console.log("mod2 init");},},
];
for (let mod of mods) {// mod.init(); // 不好,只要有一個mod的init方法出錯,影響后續try {mod.init(); // 'mod2 init',mod2照常運行} catch (error) {console.log(error); // TypeError: Assignment to constant variable.}
}

總結 & 問點

  • 錯誤對象中的哪些屬性在全部瀏覽器中都向用戶顯示?
  • finally 對 try 或 catch 中的非 return 語句和 return 語句分別有什么影響?
  • 請舉例說明有哪些常見錯誤類型及其出現的原因
  • 請寫一段代碼,在 try/catch 塊中,確定錯誤的類型
  • throw 操作符必須有值嘛?需要什么數據類型的值?如何才能既使用該操作符又不影響后續代碼執行?
  • 寫一段代碼,通過繼承 Error 創建一個自定義的錯誤類型,創建其實例并用 throw 將其拋出
  • 寫一段代碼,在一個函數里,通過創建自定義錯誤類型提高其可維護性
  • 常見的類型轉換錯誤有哪些?分別該如何避免呢?
  • 應分別怎樣檢測,以避免原始值和對象值在函數傳參時可能發生的數據類型錯誤?
  • 寫一個方法,處理與服務器通信的 url 的正確格式
  • 寫一段代碼,用 for 循環模擬模塊初始化,某一模塊加載時發生錯誤但不影響后續模塊

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/717001.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/717001.shtml
英文地址,請注明出處:http://en.pswp.cn/news/717001.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

vsomeip源碼剖析--00環境搭建

環境 Win11 WSL2 Ubuntu22.04安裝依賴 sudo apt-get install cmake sudo apt-get install libboost-system1.71-dev libboost-thread1.71-dev libboost-log1.71-dev源碼編譯 獲取源碼 https://github.com/COVESA/vsomeip.git編譯 cd vsomeip mkdir build cd build// 一般…

漫漫數學之旅035

文章目錄 經典格言數學習題古今評注名人小傳 - 黎勒?笛卡爾 經典格言 完美的數和完美的人是同樣罕見的。——黎勒?笛卡爾&#xff08;Ren Descrates&#xff09; 完美的數和完美的人都是極為罕見的。這句話表達了一個哲學觀點&#xff0c;即無論是在數學領域還是人類自身&am…

Spring框架相關問題

RabbitMQ相關問題 Spring框架相關問題 一、Spring容器中的Bean是線程安全的嗎&#xff1f;二、如何保證Spring容器中的Bean是線程安全的呢&#xff1f;三、什么情況下會觸發Spring事務回滾&#xff1f;四、如果事務方法拋出IOException&#xff0c;是否會觸發Spring事務回滾&a…

Zookeeper學習2:原理、常用腳本、選舉機制、監聽器

文章目錄 原理選舉機制&#xff08;重點&#xff09;情況1&#xff1a;正常啟動集群情況2&#xff1a;集群啟動完&#xff0c;中途有機器掛了 監聽器客戶端向服務端寫入數據客戶端向服務端Leader節點寫入客戶端向服務端Follower節點寫入 Paxos算法&#xff08;每個節點都可以提…

AMDGPU KFD Test 編譯使用

ROCT-Thunk-Interface是一個用于在ROCm軟件堆棧中提供設備無關性的層。它是ROCm的一部分,允許不同的硬件平臺(如AMD GPU和Intel CPU)使用相同的API進行計算。 要安裝ROCT-Thunk-Interface,首先需要創建一個新的目錄,并進入該目錄: mkdir rocm-build cd rocm-build然后,…

ng : 無法將ng項識別為 cmdlet、函數、腳本文件或可運行程序的名稱

ng : 無法將“ng”項識別為 cmdlet、函數、腳本文件或可運行程序的名稱”&#xff0c;出現這種錯誤&#xff0c;那說明你angular-cli沒有下載所以環境變量里沒有相應的東西 1、需要在cmd里輸入npm install -g angular/cli 2、之后運行angular命令時還可能出現這種錯誤 “ng : …

ruoyi 圖片等文件資源讀取

老是忘&#xff0c;記錄一下 ResourcesConfig 文件下 /** 本地文件上傳路徑 */ registry.addResourceHandler(Constants.RESOURCE_PREFIX "/**").addResourceLocations("file:" RuoYiConfig.getProfile() "/"); /*** 資源映射路徑 前綴*/ …

kafka消費者重平衡是什么?怎么避免?

消費者重平衡是指主題下的分區怎么分配給消費者的過程。下面這個圖可以看出該過程&#xff1a;原來有2個消費者&#xff0c;3個分區&#xff0c;其中一個消費者肯定就的處理2個分區了。那么當新加入消費者時&#xff0c;則每個消費者就只處理一個分區了。處理這個分區過程的叫協…

詳解Nacos注冊中心的使用

文章目錄 1、安裝2、服務注冊2.1、引入依賴2.2、配置nacos地址2.3、重啟 3、服務分級存儲模型3.1、給user-service配置集群3.2、同集群優先的負載均衡 4、權重配置5、環境隔離5.1、創建namespace5.2、配置namespace 6、Nacos與Eureka的區別7、代碼免費分享 ?&#x1f343;作者…

首例以“冠狀病毒”為主題的勒索病毒,篡改系統MBR

前言概述 2020年勒索病毒攻擊仍然是網絡安全的最大威脅&#xff0c;在短短三個月的時間里&#xff0c;已經出現了多款新型的勒索病毒&#xff0c;關于2020年勒索病毒攻擊新趨勢&#xff0c;可以閱讀筆者寫的上一篇文章&#xff0c;里面有詳細的分析&#xff0c;從目前觀察到的…

Linux 學習筆記(9)

九、 運行級別 1 、 Linux 系統的運行級別 (runlevel) Linux 系統有 7 個運行級別&#xff0c; Linux 系統任何時候都運行在一個指定的運行級別上&#xff0c;不同的運行級 別所運行的程序和服務不盡相同&#xff0c;所要完成的工作和要達到的目的也不相同 運行級別…

RH850P1X芯片學習筆記-Generic Timer Module -ATOM

文章目錄 ARU-connected Timer Output Module (ATOM)OverviewGLOBAL CHANNEL CONTROL BLOCK ATOM Channel architectureATOM Channel modesSOMP-Signal Output Mode PWMSOMP - ARUSOMC-Signal Output Mode CompareSOMC - ARUSOMC – COMPARE COMMANDSOMC – OUTPUT ACTIONATOM …

Python縮進規則

Python的縮進規則是Python語法中非常重要的一部分&#xff0c;也是Python語言獨特的特點之一。在Python中&#xff0c;縮進被用來表示代碼塊的層次結構&#xff0c;而不是像其他語言一樣使用大括號或關鍵詞。這種縮進規則使得Python代碼更加簡潔、易讀、易于理解&#xff0c;同…

python模塊百科_操作系統接口_os【一】

python模塊百科_操作系統接口_os【一】 os --- 多種操作系統接口一、相關模塊1.1 os.path 文件路徑1.2 fileinput 文件讀取1.3 tempfile 臨時文件和目錄1.4 shutil 高級文件和目錄1.5 platform 操作系統底層模塊 二、關于函數適用性的說明2.1 與操作系統相同的接口2.2 支持字節…

Git版本管理常用指令

Git常用命令 一、基本指令二、本地倉庫管理三、遠程倉庫管理四、分支管理五、儲藏區六、標簽管理一、基本指令 查看Git安裝版本:git --version 查看log指令的幫助信息:git log --help 配置Git用戶名:git config --global user.name “xxxxx” 配置Git郵箱: git config --…

2024年騰訊云新用戶優惠券領取入口及使用教程

隨著云計算技術的不斷發展和普及&#xff0c;越來越多的個人和企業選擇使用云服務。騰訊云作為國內領先的云服務提供商&#xff0c;為了吸引新用戶&#xff0c;經常推出各種優惠活動&#xff0c;其中就包括新用戶專屬優惠券&#xff0c;本文將為大家分享騰訊云新用戶優惠券的領…

5個好玩神奇還免費的工具網站收藏不后悔-搜嗖工具箱

生命倒計時 http://www.thismuchlonger.com 這是一個相哇塞的網站&#xff0c;可以讓我們靜下心來好好想想我們來這個世界究竟為了什么&#xff0c;因為當我們作為命運的主宰者。敲打鍵盤設定好自己一生長度的時候&#xff0c;我們的剩余壽命已經成絕對值&#xff0c;一旦生命…

創建型模式之原型模式

一、概述 1、工作原理&#xff1a;將一個原型對象傳給要發動創建的對象(即客戶端對象),這個要發動創建的對象通過請求原型對象復制自己來實現創建過程 2、通過克隆方法所創建的對象是全新的對象&#xff0c;它們在內存中擁有新的地址&#xff0c;每一個克隆對象都是獨立的 3…

MySQL 中的 varchar 和 char 有什么區別?MySQL中 in 和 exists 區別?

MySQL 中的 varchar 和 char 有什么區別&#xff1f; char 是一個定長字段,假如申請了char(10)的空間,那么無論實際存儲多少內容.該字段都占用 10 個字符,而 varchar 是變長的,也就是說申請的只是最大長度,占用的空間為實際字符長度1,最后一個字符存儲使用了多長的空間. 在檢索…

李沐動手學習深度學習——3.6練習

本節直接實現了基于數學定義softmax運算的softmax函數。這可能會導致什么問題&#xff1f;提示&#xff1a;嘗試計算exp(50)的大小。 可能存在超過計算機最大64位的存儲&#xff0c;導致精度溢出&#xff0c;影響最終計算結果。 本節中的函數cross_entropy是根據交叉熵損失函數…