偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

JavaScript錯(cuò)誤處理和堆棧追蹤淺析

開發(fā) 前端
有時(shí)我們會(huì)忽略錯(cuò)誤處理和堆棧追蹤的一些細(xì)節(jié), 但是這些細(xì)節(jié)對(duì)于寫與測(cè)試或錯(cuò)誤處理相關(guān)的庫(kù)來(lái)說(shuō)是非常有用的。合理地處理堆棧信息能使你清除無(wú)用的數(shù)據(jù), 而只專注于有用的數(shù)據(jù). 同時(shí), 當(dāng)更好地理解 Errors 對(duì)象及其相關(guān)屬性之后, 能有助于你更充分地利用 Errors。

[[187726]]

有時(shí)我們會(huì)忽略錯(cuò)誤處理和堆棧追蹤的一些細(xì)節(jié), 但是這些細(xì)節(jié)對(duì)于寫與測(cè)試或錯(cuò)誤處理相關(guān)的庫(kù)來(lái)說(shuō)是非常有用的. 例如這周, 對(duì)于 Chai 就有一個(gè)非常棒的PR, 該P(yáng)R極大地改善了我們處理堆棧的方式, 當(dāng)用戶的斷言失敗的時(shí)候, 我們會(huì)給予更多的提示信息(幫助用戶進(jìn)行定位).

合理地處理堆棧信息能使你清除無(wú)用的數(shù)據(jù), 而只專注于有用的數(shù)據(jù). 同時(shí), 當(dāng)更好地理解 Errors 對(duì)象及其相關(guān)屬性之后, 能有助于你更充分地利用 Errors.

(函數(shù)的)調(diào)用棧是怎么工作的

在談?wù)撳e(cuò)誤之前, 先要了解下(函數(shù)的)調(diào)用棧的原理:

當(dāng)有一個(gè)函數(shù)被調(diào)用的時(shí)候, 它就被壓入到堆棧的頂部, 該函數(shù)運(yùn)行完成之后, 又會(huì)從堆棧的頂部被移除.

堆棧的數(shù)據(jù)結(jié)構(gòu)就是后進(jìn)先出, 以 LIFO (last in, first out) 著稱.

例如:

  1. function c() { 
  2.  
  3.     console.log('c'); 
  4.  
  5.  
  6.   
  7.  
  8. function b() { 
  9.  
  10.     console.log('b'); 
  11.  
  12.     c(); 
  13.  
  14.  
  15.   
  16.  
  17. function a() { 
  18.  
  19.     console.log('a'); 
  20.  
  21.     b(); 
  22.  
  23.  
  24.   
  25.  
  26. a();  

在上述的示例中, 當(dāng)函數(shù) a 運(yùn)行時(shí), 其會(huì)被添加到堆棧的頂部. 然后, 當(dāng)函數(shù) b 在函數(shù) a 的內(nèi)部被調(diào)用時(shí), 函數(shù) b 會(huì)被壓入到堆棧的頂部. 當(dāng)函數(shù) c 在函數(shù) b 的內(nèi)部被調(diào)用時(shí)也會(huì)被壓入到堆棧的頂部.

當(dāng)函數(shù) c 運(yùn)行時(shí), 堆棧中就包含了 a, b 和 c(按此順序).

當(dāng)函數(shù) c 運(yùn)行完畢之后, 就會(huì)從堆棧的頂部被移除, 然后函數(shù)調(diào)用的控制流就回到函數(shù) b. 函數(shù) b 運(yùn)行完之后, 也會(huì)從堆棧的頂部被移除, 然后函數(shù)調(diào)用的控制流就回到函數(shù) a. ***, 函數(shù) a 運(yùn)行完成之后也會(huì)從堆棧的頂部被移除.

為了更好地在demo中演示堆棧的行為, 可以使用 console.trace() 在控制臺(tái)輸出當(dāng)前的堆棧數(shù)據(jù). 同時(shí), 你要以從上至下的順序閱讀輸出的堆棧數(shù)據(jù).

  1. function c() { 
  2.  
  3.     console.log('c'); 
  4.  
  5.     console.trace(); 
  6.  
  7.  
  8.   
  9.  
  10. function b() { 
  11.  
  12.     console.log('b'); 
  13.  
  14.     c(); 
  15.  
  16.  
  17.   
  18.  
  19. function a() { 
  20.  
  21.     console.log('a'); 
  22.  
  23.     b(); 
  24.  
  25.  
  26.   
  27. a();  

在 Node 的 REPL 模式中運(yùn)行上述代碼會(huì)得到如下輸出:

  1. Trace 
  2.  
  3.     at c (repl:3:9) 
  4.  
  5.     at b (repl:3:1) 
  6.  
  7.     at a (repl:3:1) 
  8.  
  9.     at repl:1:1 // <-- For now feel free to ignore anything below this point, these are Node's internals 
  10.  
  11.     at realRunInThisContextScript (vm.js:22:35) 
  12.  
  13.     at sigintHandlersWrap (vm.js:98:12) 
  14.  
  15.     at ContextifyScript.Script.runInThisContext (vm.js:24:12) 
  16.  
  17.     at REPLServer.defaultEval (repl.js:313:29) 
  18.  
  19.     at bound (domain.js:280:14) 
  20.  
  21.     at REPLServer.runBound [as eval] (domain.js:293:12)  

正如所看到的, 當(dāng)從函數(shù) c 中輸出時(shí), 堆棧中包含了函數(shù) a, b 以及c.

如果在函數(shù) c 運(yùn)行完成之后, 在函數(shù) b 中輸出當(dāng)前的堆棧數(shù)據(jù), 就會(huì)看到函數(shù) c 已經(jīng)從堆棧的頂部被移除, 此時(shí)堆棧中僅包括函數(shù) a 和 b.

  1. function c() { 
  2.  
  3.     console.log('c'); 
  4.  
  5.  
  6.   
  7.  
  8. function b() { 
  9.  
  10.     console.log('b'); 
  11.  
  12.     c(); 
  13.  
  14.     console.trace(); 
  15.  
  16.  
  17.   
  18.  
  19. function a() { 
  20.  
  21.     console.log('a'); 
  22.  
  23.     b(); 
  24.  
  25.  

正如所看到的, 函數(shù) c 運(yùn)行完成之后, 已經(jīng)從堆棧的頂部被移除.

  1. Trace 
  2.  
  3.     at b (repl:4:9) 
  4.  
  5.     at a (repl:3:1) 
  6.  
  7.     at repl:1:1  // <-- For now feel free to ignore anything below this point, these are Node's internals 
  8.  
  9.     at realRunInThisContextScript (vm.js:22:35) 
  10.  
  11.     at sigintHandlersWrap (vm.js:98:12) 
  12.  
  13.     at ContextifyScript.Script.runInThisContext (vm.js:24:12) 
  14.  
  15.     at REPLServer.defaultEval (repl.js:313:29) 
  16.  
  17.     at bound (domain.js:280:14) 
  18.  
  19.     at REPLServer.runBound [as eval] (domain.js:293:12) 
  20.  
  21.     at REPLServer.onLine (repl.js:513:10)  

Error對(duì)象和錯(cuò)誤處理

當(dāng)程序運(yùn)行出現(xiàn)錯(cuò)誤時(shí), 通常會(huì)拋出一個(gè) Error 對(duì)象. Error 對(duì)象可以作為用戶自定義錯(cuò)誤對(duì)象繼承的原型.

Error.prototype 對(duì)象包含如下屬性:

  • constructor–指向?qū)嵗臉?gòu)造函數(shù)
  • message–錯(cuò)誤信息
  • name–錯(cuò)誤的名字(類型)

上述是 Error.prototype 的標(biāo)準(zhǔn)屬性, 此外, 不同的運(yùn)行環(huán)境都有其特定的屬性. 在例如 Node, Firefox, Chrome, Edge, IE 10+, Opera 以及 Safari 6+ 這樣的環(huán)境中, Error 對(duì)象具備 stack 屬性, 該屬性包含了錯(cuò)誤的堆棧軌跡. 一個(gè)錯(cuò)誤實(shí)例的堆棧軌跡包含了自構(gòu)造函數(shù)之后的所有堆棧結(jié)構(gòu).

如果想了解更多關(guān)于 Error 對(duì)象的特定屬性, 可以閱讀 MDN 上的這篇文章.

為了拋出一個(gè)錯(cuò)誤, 必須使用 throw 關(guān)鍵字. 為了 catch 一個(gè)拋出的錯(cuò)誤, 必須使用 try...catch 包含可能跑出錯(cuò)誤的代碼. Catch的參數(shù)是被跑出的錯(cuò)誤實(shí)例.

如 Java 一樣, JavaScript 也允許在 try/catch 之后使用 finally 關(guān)鍵字. 在處理完錯(cuò)誤之后, 可以在finally 語(yǔ)句塊作一些清除工作.

在語(yǔ)法上, 你可以使用 try 語(yǔ)句塊而其后不必跟著 catch 語(yǔ)句塊, 但必須跟著 finally 語(yǔ)句塊. 這意味著有三種不同的 try 語(yǔ)句形式:

  • try...catch
  • try...finally
  • try...catch...finally

Try語(yǔ)句內(nèi)還可以在嵌入 try 語(yǔ)句:

  1. try { 
  2.  
  3.     try { 
  4.  
  5.         throw new Error('Nested error.'); // The error thrown here will be caught by its own `catch` clause 
  6.  
  7.     } catch (nestedErr) { 
  8.  
  9.         console.log('Nested catch'); // This runs 
  10.  
  11.     } 
  12.  
  13. } catch (err) { 
  14.  
  15.     console.log('This will not run.'); 
  16.  
  17.  

也可以在 catch 或 finally 中嵌入 try 語(yǔ)句:

  1. try { 
  2.  
  3.     console.log('The try block is running...'); 
  4.  
  5. } finally { 
  6.  
  7.     try { 
  8.  
  9.         throw new Error('Error inside finally.'); 
  10.  
  11.     } catch (err) { 
  12.  
  13.         console.log('Caught an error inside the finally block.'); 
  14.  
  15.     } 
  16.  
  17.  

需要重點(diǎn)說(shuō)明一下的是在拋出錯(cuò)誤時(shí), 可以只拋出一個(gè)簡(jiǎn)單值而不是 Error 對(duì)象. 盡管這看起來(lái)看酷并且是允許的, 但這并不是一個(gè)推薦的做法, 尤其是對(duì)于一些需要處理他人代碼的庫(kù)和框架的開發(fā)者, 因?yàn)闆](méi)有標(biāo)準(zhǔn)可以參考, 也無(wú)法得知會(huì)從用戶那里得到什么. 你不能信任用戶會(huì)拋出 Error 對(duì)象, 因?yàn)樗麄兛赡懿粫?huì)這么做, 而是簡(jiǎn)單的拋出一個(gè)字符串或者數(shù)值. 這也意味著很難去處理堆棧信息和其它元信息.

例如:

  1. function runWithoutThrowing(func) { 
  2.  
  3.     try { 
  4.  
  5.         func(); 
  6.  
  7.     } catch (e) { 
  8.  
  9.         console.log('There was an error, but I will not throw it.'); 
  10.  
  11.         console.log('The error\'s message was: ' + e.message) 
  12.  
  13.     } 
  14.  
  15.  
  16.   
  17.  
  18. function funcThatThrowsError() { 
  19.  
  20.     throw new TypeError('I am a TypeError.'); 
  21.  
  22.  
  23.   
  24.  
  25. runWithoutThrowing(funcThatThrowsError);  

如果用戶傳遞給函數(shù) runWithoutThrowing 的參數(shù)拋出了一個(gè)錯(cuò)誤對(duì)象, 上面的代碼能正常捕獲錯(cuò)誤. 然后, 如果是拋出一個(gè)字符串, 就會(huì)碰到一些問(wèn)題了:

  1. function runWithoutThrowing(func) { 
  2.  
  3.     try { 
  4.  
  5.         func(); 
  6.  
  7.     } catch (e) { 
  8.  
  9.         console.log('There was an error, but I will not throw it.'); 
  10.  
  11.         console.log('The error\'s message was: ' + e.message) 
  12.  
  13.     } 
  14.  
  15.  
  16.   
  17.  
  18. function funcThatThrowsString() { 
  19.  
  20.     throw 'I am a String.'
  21.  
  22.  
  23.   
  24.  
  25. runWithoutThrowing(funcThatThrowsString);  

現(xiàn)在第二個(gè) console.log 會(huì)輸出undefined. 這看起來(lái)不是很重要, 但如果你需要確保 Error 對(duì)象有一個(gè)特定的屬性或者用另一種方式來(lái)處理 Error 對(duì)象的特定屬性(例如 Chai的throws斷言的做法), 你就得做大量的工作來(lái)確保程序的正確運(yùn)行.

同時(shí), 如果拋出的不是 Error 對(duì)象, 也就獲取不到 stack 屬性.

Errors 也可以被作為其它對(duì)象, 你也不必拋出它們, 這也是為什么大多數(shù)回調(diào)函數(shù)把 Errors 作為***個(gè)參數(shù)的原因. 例如:

  1. const fs = require('fs'); 
  2.  
  3.   
  4.  
  5. fs.readdir('/example/i-do-not-exist'function callback(err, dirs) { 
  6.  
  7.     if (err instanceof Error) { 
  8.  
  9.         // `readdir` will throw an error because that directory does not exist 
  10.  
  11.         // We will now be able to use the error object passed by it in our callback function 
  12.  
  13.         console.log('Error Message: ' + err.message); 
  14.  
  15.         console.log('See? We can use Errors without using try statements.'); 
  16.  
  17.     } else { 
  18.  
  19.         console.log(dirs); 
  20.  
  21.     } 
  22.  
  23. });  

***, Error 對(duì)象也可以用于 rejected promise, 這使得很容易處理 rejected promise:

  1. new Promise(function(resolve, reject) { 
  2.  
  3.     reject(new Error('The promise was rejected.')); 
  4.  
  5. }).then(function() { 
  6.  
  7.     console.log('I am an error.'); 
  8.  
  9. }).catch(function(err) { 
  10.  
  11.     if (err instanceof Error) { 
  12.  
  13.         console.log('The promise was rejected with an error.'); 
  14.  
  15.         console.log('Error Message: ' + err.message); 
  16.  
  17.     } 
  18.  
  19. });  

處理堆棧

這一節(jié)是針對(duì)支持 Error.captureStackTrace的運(yùn)行環(huán)境, 例如Nodejs.

Error.captureStackTrace 的***個(gè)參數(shù)是 object, 第二個(gè)可選參數(shù)是一個(gè) function.Error.captureStackTrace 會(huì)捕獲堆棧信息, 并在***個(gè)參數(shù)中創(chuàng)建 stack 屬性來(lái)存儲(chǔ)捕獲到的堆棧信息. 如果提供了第二個(gè)參數(shù), 該函數(shù)將作為堆棧調(diào)用的終點(diǎn). 因此, 捕獲到的堆棧信息將只顯示該函數(shù)調(diào)用之前的信息.

用下面的兩個(gè)demo來(lái)解釋一下. ***個(gè), 僅將捕獲到的堆棧信息存于一個(gè)普通的對(duì)象之中:

  1. const myObj = {}; 
  2.  
  3.   
  4.  
  5. function c() { 
  6.  
  7.  
  8.   
  9.  
  10. function b() { 
  11.  
  12.     // Here we will store the current stack trace into myObj 
  13.  
  14.     Error.captureStackTrace(myObj); 
  15.  
  16.     c(); 
  17.  
  18.  
  19.   
  20.  
  21. function a() { 
  22.  
  23.     b(); 
  24.  
  25.  
  26.   
  27.  
  28. // First we will call these functions 
  29.  
  30. a(); 
  31.  
  32.   
  33.  
  34. // Now let's see what is the stack trace stored into myObj.stack 
  35.  
  36. console.log(myObj.stack); 
  37.  
  38.   
  39.  
  40. // This will print the following stack to the console: 
  41.  
  42. //    at b (repl:3:7) <-- Since it was called inside B, the B call is the last entry in the stack 
  43.  
  44. //    at a (repl:2:1) 
  45.  
  46. //    at repl:1:1 <-- Node internals below this line 
  47.  
  48. //    at realRunInThisContextScript (vm.js:22:35) 
  49.  
  50. //    at sigintHandlersWrap (vm.js:98:12) 
  51.  
  52. //    at ContextifyScript.Script.runInThisContext (vm.js:24:12) 
  53.  
  54. //    at REPLServer.defaultEval (repl.js:313:29) 
  55.  
  56. //    at bound (domain.js:280:14) 
  57.  
  58. //    at REPLServer.runBound [as eval] (domain.js:293:12) 
  59.  
  60. //    at REPLServer.onLine (repl.js:513:10)  

從上面的示例可以看出, 首先調(diào)用函數(shù) a(被壓入堆棧), 然后在 a 里面調(diào)用函數(shù) b(被壓入堆棧且在a之上), 然后在 b 中捕獲到當(dāng)前的堆棧信息, 并將其存儲(chǔ)到 myObj 中. 所以, 在控制臺(tái)輸出的堆棧信息中僅包含了 a和 b 的調(diào)用信息.

現(xiàn)在, 我們給 Error.captureStackTrace 傳遞一個(gè)函數(shù)作為第二個(gè)參數(shù), 看下輸出信息:

  1. const myObj = {}; 
  2.  
  3.   
  4.  
  5. function d() { 
  6.  
  7.     // Here we will store the current stack trace into myObj 
  8.  
  9.     // This time we will hide all the frames after `b` and `b` itself 
  10.  
  11.     Error.captureStackTrace(myObj, b); 
  12.  
  13.  
  14.   
  15.  
  16. function c() { 
  17.  
  18.     d(); 
  19.  
  20.  
  21.   
  22.  
  23. function b() { 
  24.  
  25.     c(); 
  26.  
  27.  
  28.   
  29.  
  30. function a() { 
  31.  
  32.     b(); 
  33.  
  34.  
  35.   
  36.  
  37. // First we will call these functions 
  38.  
  39. a(); 
  40.  
  41.   
  42.  
  43. // Now let's see what is the stack trace stored into myObj.stack 
  44.  
  45. console.log(myObj.stack); 
  46.  
  47.   
  48.  
  49. // This will print the following stack to the console: 
  50.  
  51. //    at a (repl:2:1) <-- As you can see here we only get frames before `b` was called 
  52.  
  53. //    at repl:1:1 <-- Node internals below this line 
  54.  
  55. //    at realRunInThisContextScript (vm.js:22:35) 
  56.  
  57. //    at sigintHandlersWrap (vm.js:98:12) 
  58.  
  59. //    at ContextifyScript.Script.runInThisContext (vm.js:24:12) 
  60.  
  61. //    at REPLServer.defaultEval (repl.js:313:29) 
  62.  
  63. //    at bound (domain.js:280:14) 
  64.  
  65. //    at REPLServer.runBound [as eval] (domain.js:293:12) 
  66.  
  67. //    at REPLServer.onLine (repl.js:513:10) 
  68.  
  69. //    at emitOne (events.js:101:20)  

當(dāng)將函數(shù) b 作為第二個(gè)參數(shù)傳給 Error.captureStackTraceFunction 時(shí), 輸出的堆棧就只包含了函數(shù) b 調(diào)用之前的信息(盡管 Error.captureStackTraceFunction 是在函數(shù) d 中調(diào)用的), 這也就是為什么只在控制臺(tái)輸出了 a. 這樣處理方式的好處就是用來(lái)隱藏一些與用戶無(wú)關(guān)的內(nèi)部實(shí)現(xiàn)細(xì)節(jié). 

責(zé)任編輯:龐桂玉 來(lái)源: 前端大全
相關(guān)推薦

2017-03-08 08:57:04

JavaScript錯(cuò)誤堆棧

2017-04-25 15:30:23

堆棧函數(shù)JavaScript

2022-11-16 08:41:43

2024-03-27 08:18:02

Spring映射HTML

2023-11-08 15:04:55

事務(wù)GORM

2021-05-11 10:01:54

avaScript錯(cuò)誤處理

2020-09-15 08:28:17

JavaScript錯(cuò)誤處理

2021-04-14 07:08:14

Nodejs錯(cuò)誤處理

2020-09-14 08:35:36

JavaScript編程開發(fā)

2014-11-17 10:05:12

Go語(yǔ)言

2009-08-05 16:04:50

2010-06-01 16:14:04

2021-04-29 09:02:44

語(yǔ)言Go 處理

2023-10-28 16:30:19

Golang開發(fā)

2023-10-26 15:49:53

Go日志

2022-05-06 08:00:51

Golang編程語(yǔ)言Java

2014-07-30 09:56:41

iPhoneiPad

2016-08-19 10:41:42

Swift 2錯(cuò)誤

2023-12-26 22:05:53

并發(fā)代碼goroutines

2015-08-19 14:11:56

SQL Server錯(cuò)誤處理
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)