欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

簡單了解TypeScript中如何繼承 Error 類

 更新時間:2019年06月21日 17:03:56   作者:Harttle  
這篇文章主要介紹了簡單了解TypeScript中如何繼承 Error 類,一個典型的網(wǎng)站服務器可能需要有 NetworkError, DatabaseError, UnauthorizedError 等。 我們希望這些類都擁有 Error 的特性:有錯誤消息、有調(diào)用棧、有方便打印的 toString 等。,需要的朋友可以參考下

前言

在JavaScript 中很多時候都需要自定義錯誤,尤其是開發(fā) Node.js 應用的時候。 比如一個典型的網(wǎng)站服務器可能需要有 NetworkError, DatabaseError, UnauthorizedError 等。 我們希望這些類都擁有 Error 的特性:有錯誤消息、有調(diào)用棧、有方便打印的 toString 等。 最直觀的實現(xiàn)方式便是 繼承 Error 類。 但考慮 TypeScript 需要編譯到 ES5 兼容性問題會較為復雜, 本文用來幫助理解 TypeScript 中繼承 Error 的問題來源以及對應的幾種解決方式。

我們需要怎樣的 CustomError

為了容易討論最佳實踐,首先明確我們自定義的 CustomError 需要做到哪些功能。 下面是 Harttle 的觀點:

  1. 可以調(diào)用 new CustomError() 來創(chuàng)建,并且 instanceof Error 操作應該返回 true??梢杂脕韯?chuàng)建是基本要求,能夠被視為 Error 的實例能夠兼容既有系統(tǒng)(比如 toString() 要返回調(diào)用棧),同時符合慣例。
  2. .stack 屬性首行應為 CustomeError: <message>。如果是 Error: <message> 可能就沒那么漂亮。
  3. .stack 屬性應當包含調(diào)用棧并指向 new CustomError() 的那一行。這一點可能是關鍵,如果指向 CustomError 構(gòu)造函數(shù)中的某一行,就會給這個類的使用方造成困惑。

下面舉個例子,這是一個 message 為 "intended" 的 CustomError 的 .stack 屬性值:

CustomError: intended
at Object.<anonymous> (/Users/harttle/Downloads/bar/a.js:10:13)
at Module._compile (module.js:653:30)
at Object.Module._extensions..js (module.js:664:10)
at Module.load (module.js:566:32)
at tryModuleLoad (module.js:506:12)
at Function.Module._load (module.js:498:3)
at Function.Module.runMain (module.js:694:10)
at startup (bootstrap_node.js:204:16)
at bootstrap_node.js:625:3

ES5 中如何繼承 Error?

Error 是一個特殊的對象,或者說 JavaScript 的 new 是一個奇葩的存在。 為方便后續(xù)討論,我們先討論組 ES5 時代是怎樣繼承 Error 的。 我們說 JavaScript 是一門混雜的語言,如何繼承 Error 就是一個典型的例子。 如果你熟悉 原型繼承的方式,應該會寫出如下代碼:

function CustomError (message) {
Error.call(this, message)
}
CustomError.prototype = new Error()

因為 stack 只在 new 的時候生成,上述實現(xiàn)不能滿足功能 2 和功能 3,也就是說:

  • stack 的第一行是總是 Error 而不是 CustomError 且不包含 message 信息。
  • stack 總是指向 new Error() 的那一行,而不是 new CustomError()。

Node 文檔 中描述了一個 captureStackTrace 方法來解決這個問題,改動后的實現(xiàn)如下:

function CustomError (msg) {
this.name = 'CustomError'
this.message = msg
Error.captureStackTrace(this, CustomError)
}
CustomError.prototype = new Error()

其中 .captureStackTrace() 會使用傳入對象的 name 和 message 來生成 stack 的前綴;同時第二個參數(shù)用來指定在調(diào)用棧中忽略掉哪一部分,這樣棧就會指向 new CustomError 的地方而不是 captureStackTrace() 的地方。

ES6 中如何繼承 Error?

既然 ES6 通過 class 和 extends 等關鍵字給出了類繼承機制, 那么想必通過編寫 CustomError 類來繼承 Error。事實也確實如此,只需要在構(gòu)造函數(shù)中調(diào)用父類構(gòu)造函數(shù)并賦值 name 即可實現(xiàn)文章開始提到的三個功能:

class CustomError extends Error {
constructor(msg) {
super(msg)
this.name = 'CustomError'
}
}

TypeScript 中如何繼承 Error?

ES6 中提供了 new.target 屬性, 使得 Error 的構(gòu)造函數(shù)中可以獲取 CustomError 的信息,以完成原型鏈的調(diào)整。 因此 TypeScript 需要編譯到 ES5 時上述功能仍然是無法自動實現(xiàn)。 在 TypeScript 中的體現(xiàn)是形如上述 ES6 的代碼片段會被編譯成:

var CustomError = /** @class */ (function (_super) {
__extends(CustomError, _super);
function CustomError(msg) {
var _this = _super.call(this, msg) || this;
_this.name = 'CustomError';
return _this;
}
return CustomError;
}(Error));

注意 var _this = _super.call(this, msg) || this; 中 this 被替換掉了。 在 TypeScript 2.1 的 changelog 中描述了這個 Breaking Change。 **這會造成 CustomError 的所有對象方法都無法使用,這里介紹幾種 workaround:

題外話,這個分支可能會導致測試覆蓋率中的 分支未覆蓋問題??梢灾辉?ES6 下產(chǎn)生測試覆蓋報告來解決。

1. 使用 setPrototypeOf 還原原型鏈

這是 TypeScript 官方給出的解決方法,見這里

class CustomError extends Error {
constructor(message) {
super(message);
Object.setPrototypeOf(this, FooError.prototype);
}
}

注意這是一個性能很差的方法,且在 ES6 中提出,兼容性也很差。在不兼容的環(huán)境下可以使用 __proto__ 來替代。

2. 堅持使用 ES5 的方式

不使用 ES6 特性,仍然使用本文前面介紹的 『ES5 中如何繼承 Error?』給出的方法。

3. 限制對象方法的使用

雖然 CustomError 的對象函數(shù)無法使用,但 CustomError 仍然支持 protected 級別的方法供子類使用,閹割的地方在于自己不能調(diào)用。 由于 JavaScript 中對象屬性必須在構(gòu)造函數(shù)內(nèi)賦值,因此對象屬性也不會受到影響。也就是說:

class CustomError extends Error {
count: number = 0
constructor(msg) {
super(msg)
this.count // OK,屬性不受影響
this.print() // TypeError: _this.print is not a function,因為 this 被替換了
}
print() { 
console.log(this.stack)
}
}
class DerivedError extends CustomError {
constructor(msg) {
super(msg)
super.print() // OK,因為 print 是直接從父類原型獲取的,即 `_super.prototype.print`
}
}

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關文章

  • JS添加刪除一組文本框并對輸入信息加以驗證判斷其正確性

    JS添加刪除一組文本框并對輸入信息加以驗證判斷其正確性

    需要添加幾組數(shù)據(jù)到數(shù)據(jù)庫,但是具體幾組數(shù)據(jù)不確定,有客戶來填寫,所以,這里我用JS進行添加刪除子方案,并要對方案輸入的正確性加以判斷,感興趣的朋友可以了解下
    2013-04-04
  • jquery的$getjson調(diào)用并獲取遠程的JSON字符串問題

    jquery的$getjson調(diào)用并獲取遠程的JSON字符串問題

    jQuery中常用getJSON來調(diào)用并獲取遠程的JSON字符串,將其轉(zhuǎn)換為JSON對象,如果成功,則執(zhí)行回調(diào)函數(shù),本文將詳細介紹,需要的朋友可以參考下
    2012-12-12
  • js實現(xiàn)文本框中焦點在最后位置

    js實現(xiàn)文本框中焦點在最后位置

    本篇文章主要是對js實現(xiàn)文本框中焦點在最后位置的示例代碼進行了介紹,需要的朋友可以過來參考下,希望對大家有所 幫助
    2014-03-03
  • JS如何通過視頻鏈接獲取視頻時長

    JS如何通過視頻鏈接獲取視頻時長

    這篇文章主要介紹了JS如何通過視頻鏈接獲取視頻時長,這個函數(shù)用提供的URL創(chuàng)建一個新的Video元素,并在loadedmetadata事件被觸發(fā)時解析一個帶有視頻持續(xù)時間的Promise,感興趣的朋友跟隨小編一起看看吧
    2024-06-06
  • 原生js生成圖片驗證碼

    原生js生成圖片驗證碼

    這篇文章主要為大家詳細介紹了原生js生成圖片驗證碼,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-10-10
  • JavaScript擴展運算符的學習及應用詳情(ES6)

    JavaScript擴展運算符的學習及應用詳情(ES6)

    這篇文章主要介紹了JavaScript擴展運算符的學習及應用詳情(ES6),擴展運算符是ES6新增的一種運算符,他可以幫助我們簡化代碼,簡化操作,具體相關知識感興趣的小伙伴可以查看下面文章的簡單介紹
    2022-08-08
  • 最全的JavaScript開發(fā)工具列表 總有一款適合你

    最全的JavaScript開發(fā)工具列表 總有一款適合你

    最全的JavaScript開發(fā)工具列表分享給你,總有一款適合你!
    2017-06-06
  • 再次談論Javascript中的this

    再次談論Javascript中的this

    javascript中的this應用非常廣泛,對js中this總是似是而非的感覺,今天小編豁然開朗,然后再次給大家談論js中的this關鍵,感興趣的朋友跟著小編一起看看吧
    2016-06-06
  • js實現(xiàn)嵌套數(shù)組重排序

    js實現(xiàn)嵌套數(shù)組重排序

    這篇文章主要為大家詳細介紹了js實現(xiàn)嵌套數(shù)組重排序,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-01-01
  • js中獲取鍵盤事件的簡單實現(xiàn)方法

    js中獲取鍵盤事件的簡單實現(xiàn)方法

    下面小編就為大家?guī)硪黄猨s中獲取鍵盤事件的簡單實現(xiàn)方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-10-10

最新評論