TypeScript實(shí)用技巧?Nominal?Typing名義類型詳解
Nominal Typing(名義類型)
概念解析
意思是給一個(gè)類型附加上一個(gè)“名義”,從而防止結(jié)構(gòu)類型在某些情況下由于類型結(jié)構(gòu)相似而被錯(cuò)用。假設(shè)有如下代碼:
interface Vector2D { x: number, y: number }; interface Vector3D { x: number, y: number, z: number }; function calc(vector: Vector2D): void; const vector: Vector3D = { x: 1, y: 1, z: 1} calc(vector) // 并沒有拋出錯(cuò)誤
看上去calc()
函數(shù)應(yīng)該只能傳入Vector2D
類型,但其實(shí)也可以傳入Vector3D
,因?yàn)楸举|(zhì)上Vector3D
是Vector2D
的子集。對于calc()
函數(shù)來說,傳入的vector
變量的類型中只要同時(shí)具有x
、y
屬性即可通過類型校驗(yàn)。
這種特性這在TS中被稱為 Structual Typing(結(jié)構(gòu)類型)。通常來說這會給我們的編碼過程帶來便利,但極端情況下,也可能不符合我們的預(yù)期。
假如嚴(yán)格規(guī)定函數(shù)只能傳入Vector2D
類型而不能傳入Vector3D
類型,那么在類型實(shí)現(xiàn)上,就可以使用 名義上的類型(Nominal Type),通過為原類型添加一個(gè)獨(dú)有標(biāo)識來區(qū)分彼此:
interface Vector2D { x: number, y: number, __type: '2d' }; interface Vector3D { x: number, y: number, z: number, __type: '3d' }; function calc(vector: Vector2D): void;
對于interface,我們可以直接為其增加標(biāo)志屬性,但 primitive types (原始類型) 要如何處理呢?答案是使用交叉類型,例如:
type Food = string & { _type: 'food' }; type Money = number & { _type: 'money' }
你可能會對最終類型有所疑問,但這樣處理之后,他們依舊是原始類型。因?yàn)閷?shí)際上原始類型最終都會解析成對應(yīng)的 WrapperType (包裹類型),例如string → String
,number → Number
,就像在JS中一樣。這意味著你可把它們當(dāng)做原始類型使用:
const money = 100 as Money; const bill = money * 1; // bill 仍然是 number 類型
雖然這樣的使用方式顯得不太優(yōu)雅,甚至有些繁瑣,但在某些情況下至少可以保證類型安全。假如你的類型系統(tǒng)中有許多 基礎(chǔ)類型單元,那可能會非常有用。
拓展應(yīng)用
這樣的類型雖然可以被當(dāng)做原始類型使用,但本質(zhì)上又不是純粹的原始類型。我們可以利用這個(gè)特性,寫出一些非常有趣和實(shí)用的類型。
例如在字面量枚舉時(shí),我們可以在限制預(yù)設(shè)值的同時(shí),使用 基于原始類型拓展出來的名義類型 進(jìn)行兜底,從而使得我們的類型能夠在具備足夠自由性的前提下,仍能享受到TypeScript的類型提示,如下:
可以看到,CustomLiteral
名義類型不但可以享受到Literal
字面量的類型提示,又能跳出枚舉的限制,使用自定義字符串。倘若我們將Literal
和原始類型string
直接交叉:
聯(lián)合類型的機(jī)制本質(zhì)上是求并集,而求并集最終得到的類型將會是更加廣泛而通用的string
,這使得我們反而使失去了字面量類型的推導(dǎo)。想要實(shí)現(xiàn)上述效果,就需要為string
類型賦予“名義”,使它不同于普通的原始類型,不再那么“廣泛”,從而在求并集的時(shí)候,不至于被string
類型徹底拿捏。附上Typescript Playground。
在Vue中的應(yīng)用
其實(shí)在Vue3源碼中也有很多 Nominal Typing 的例子,例如VNode
、Teleport
、KeepAlive
、Fragment
等等這些內(nèi)置組件,他們的定義中都有一個(gè)標(biāo)志變量用于區(qū)分。分別對應(yīng)了__is_VNode
,__isTeleport
,__isKeepAlive
,__isFragment
。下圖是KeepAlive
組件的聲明,更多組件聲明可以移步官方倉庫查閱。
如果類型聲明的位置處在函數(shù)入?yún)⑸?,為了防止與用戶定義的屬性產(chǎn)生沖突,通常會采用unique symbol
作為鍵值來構(gòu)造名義類型,例如:
以上就是TypeScript實(shí)用技巧 Nominal Typing名義類型詳解的詳細(xì)內(nèi)容,更多關(guān)于TypeScript名義類型Nominal Typing的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
網(wǎng)頁的標(biāo)準(zhǔn),IMG不支持onload標(biāo)簽怎么辦
網(wǎng)頁的標(biāo)準(zhǔn),IMG不支持onload標(biāo)簽怎么辦...2006-06-06autojs寫一個(gè)畫板實(shí)現(xiàn)AI換頭狗頭蛇
這篇文章主要為大家介紹了autojs寫一個(gè)畫板實(shí)現(xiàn)AI換頭狗頭蛇過程示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01websocket心跳重連實(shí)現(xiàn)探索(npm:websocket-heartbeat-js)
這篇文章主要為大家介紹了websocket心跳重連實(shí)現(xiàn)探索(npm:websocket-heartbeat-js),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07詳解requestAnimationFrame和setInterval該如何選擇
這篇文章主要為大家介紹了requestAnimationFrame和setInterval該如何選擇示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪<BR>2023-03-03微信小程序中頁面FOR循環(huán)和嵌套循環(huán)
這篇文章主要介紹了微信小程序中頁面FOR循環(huán)和嵌套循環(huán)的相關(guān)資料,需要的朋友可以參考下2017-06-06JS?中Json字符串+Cookie+localstorage
這篇文章主要介紹了JS?中Json字符串+Cookie+localstorage,Json主要用于前后端交互,是一種數(shù)據(jù)格式,相較于Xml,使用起來更加便捷,下面文章將對他們詳細(xì)介紹,需要的朋友可以參考一下2021-12-12