TypeScript類型斷言VS類型守衛(wèi)示例詳解
類型斷言
類型斷言有兩種寫法,分別為value as Type
和<Type>value
,它讓 TypeScript 編譯器將 value 當(dāng)作 Type 類型。類型斷言是一個編譯時特性,不進(jìn)行類型轉(zhuǎn)換,因此不會影響變量在運(yùn)行時的數(shù)據(jù)類型。如果某變量是 any 類型,但現(xiàn)在你知道它確切的數(shù)據(jù)類型,使用類型斷言能讓 IDE 有代碼提示的能力,也能讓 TypeScript 編譯器進(jìn)行類型檢查。
類型斷言的用法如下:
const age: any = 23.44; const user: any = {age: 2} // age 的 原始數(shù)據(jù)類型是 any ,在這里將它斷言為 number 類型 (age as number).toFixed(0); (<number>age).toFixed(0); // 將user斷言為非null和非undefined user!.age
除非確切的知道變量的數(shù)據(jù)類型,否則不要使用類型斷言,這是因?yàn)轭愋蛿嘌詴?TypeScript 編譯器將變量當(dāng)做指定的類型,而不管它實(shí)際的類型,在程序運(yùn)行時可能有類型錯誤。
類型守衛(wèi)
類型守衛(wèi)的工作是將數(shù)據(jù)類型收窄,比如,變量 age 的數(shù)據(jù)類型可能是 string 也可能是 number,現(xiàn)在要將 age 收窄為 string 類型,類型守衛(wèi)能實(shí)現(xiàn)它??傮w而言,有4種方式讓數(shù)據(jù)類型收窄。
使用 in 關(guān)鍵字
如果某變量的數(shù)據(jù)類型是聯(lián)合類型,并且該變量存在一個屬性絕對不會出現(xiàn)在聯(lián)合類型的其他成員類型上,那么可以使用 in 關(guān)鍵字將類型收窄。用法如下:
function printType(value: User | Student) { // grade 只存在 Student 類型上 if ('grade' in value) { // 在 if 塊中,value 的數(shù)據(jù)類型被收窄為 Student 類型 console.log('它是 Student 類型',value.name + ':' + value.grade) } else { // 在 else 塊中,value 的數(shù)據(jù)類型被收窄為 User 類型 console.log('它是 User 類型',value.name) } }
使用 instanceof 關(guān)鍵字
用 in 關(guān)鍵字收窄數(shù)據(jù)類型有局限性,必須有一個獨(dú)特的屬性作為判別符,否則,不能用 in 關(guān)鍵字收窄。如果正在處理的變量是類的實(shí)例,那么可以用 instanceof 關(guān)鍵字收窄數(shù)據(jù)類型。用法如下:
function printInfo(value: Staff | Student) { if (value instanceof Staff) { // 在 if 塊中,value 的數(shù)據(jù)類型被收窄為 Staff 類型 console.log('它是 Staff 類型') } else { // 在 else 塊中,value 的數(shù)據(jù)類型被收窄為 Student 類型 console.log('它是 Student 類型') } }
使用 typeof 關(guān)鍵字
使用 typeof 能得到值的數(shù)據(jù)類型,如果變量的類型為基本類型,如:string、number 等,使用 typeof 也能將數(shù)據(jù)類型收窄。用法如下:
function printType(value: string | number) { if (typeof value === 'string') { // value 的類型被收窄為 string console.log('value 是 string 類型') } else { // value 的類型被收窄為 number console.log('value 是 number 類型') } } // 用對象上某個字段收窄數(shù)據(jù)類型 //如果 User 類型的 name 字段為 string 類型,Student 類型的 name 字段為 number 類型 function printObjType(obj: User | Student) { if (typeof obj.name === 'string') { // value 的類型被收窄為 User } else { // value 的類型被收窄為 Student } }
自定義類型守衛(wèi)
用 in、instanceof 和 typeof 關(guān)鍵字收窄數(shù)據(jù)類型語法很簡單,但都有各自的局限性。如果它們不能當(dāng)作類型守衛(wèi),那么能使用 is 關(guān)鍵字自定義類型守衛(wèi),寫法為:parameter is Type,這種寫法被稱為類型謂詞,要將它放在函數(shù)返回值的位置。
類型謂詞所在的函數(shù)被稱為類型謂詞函數(shù),它的返回值必須為 boolean 類型。類型謂詞中的 Type可以是任意合法的 TypeScript 數(shù)據(jù)類型,parameter 可以是函數(shù)的參數(shù)也能是 this 關(guān)鍵字。
當(dāng)類型謂詞中的 parameter 是函數(shù)的參數(shù),用法如下:
function predicateStudent(value: Student | User): value is Student { return typeof (value as any).garde === 'number' } function printObjType(obj: User | Student) { if (predicateStudent(obj)) { // obj的類型被收窄為 Student } else { // value 的類型被收窄為 User } }
上述代碼中的 predicateStudent 是類型謂詞函數(shù),如果它返回 true,那么參數(shù)的數(shù)據(jù)類型被收窄為 Student。
當(dāng)類型謂詞中的 parameter 是 this 關(guān)鍵字,這種用法通常出現(xiàn)在類的方法中,用法如下:
abstract class User { name: string; constructor(name: string) { this.name = name; } isStudent(): this is Student { return this instanceof Student; } isStaff(): this is Staff { return this instanceof Staff; } } class Student extends User{ grade: number; constructor(name: string, grade: number) { super(name) this.grade = grade } } class Staff extends User { salary: number; constructor(name: string, salary: number) { super(name) this.salary = salary } } function printObjType(obj: User) { if (obj.isStaff()) { // obj 的類型被收窄為 Staff } else if (obj.isStudent()){ // obj 的類型被收窄為 Student } }
上述代碼中的 User 是抽象類,不能被實(shí)例化,Staff 和 Student 都繼承自 User。實(shí)例方法 isStaff 用于將類型收窄為 Staff,實(shí)例方法 isStudent 用于將類型收窄為 Student。
總結(jié)
在項(xiàng)目中,如果某變量有多種數(shù)據(jù)類型,不建議使用類型斷言將它斷言成特定的類型,而是使用類型守衛(wèi)根據(jù)實(shí)際情況將類型收窄,然后按不同的類型分別處理。類型守衛(wèi)能讓應(yīng)用程序的數(shù)據(jù)類型更安全,不至于程序在編譯階段不報(bào)錯,但在運(yùn)行階段報(bào)錯。
以上就是TypeScript類型斷言VS類型守衛(wèi)示例詳解的詳細(xì)內(nèi)容,更多關(guān)于TypeScript類型斷言類型守衛(wèi)的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
js實(shí)現(xiàn)手表表盤時鐘與圓周運(yùn)動
這篇文章主要為大家詳細(xì)介紹了js實(shí)現(xiàn)手表表盤時鐘與圓周運(yùn)動,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-09-09layui table 列寬百分比顯示的實(shí)現(xiàn)方法
今天小編就為大家分享一篇layui table 列寬百分比顯示的實(shí)現(xiàn)方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-09-09使用Javascript和DOM Interfaces來處理HTML
使用Javascript和DOM Interfaces來處理HTML...2006-10-10js實(shí)現(xiàn)base64文件的處理以及下載方法
Base64是一種將二進(jìn)制數(shù)據(jù)編碼為ASCII字符的編碼方式,這篇文章主要給大家介紹了關(guān)于js實(shí)現(xiàn)base64文件的處理以及下載的相關(guān)資料,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-07-07js獲取當(dāng)前年月日時分秒的方法實(shí)例(new?Date()/moment.js)
JavaScript是一種流行的編程語言,它可以用來獲取當(dāng)前年月日,這篇文章主要給大家介紹了關(guān)于js獲取當(dāng)前年月日時分秒的相關(guān)資料,分別使用的是new?Date()/moment.js,需要的朋友可以參考下2024-07-07javascript中巧用“閉包”實(shí)現(xiàn)程序的暫停執(zhí)行功能
javascript中巧用“閉包”實(shí)現(xiàn)程序的暫停執(zhí)行功能...2007-04-04原生JavaScript實(shí)現(xiàn)Ajax異步請求
ajax現(xiàn)在是一種非常流行的技術(shù),現(xiàn)在雖然可以利用JQuery或者一些第三方插件甚至微軟提供的一些控件可以方面的實(shí)現(xiàn)ajax功能,但是明白其原理也是非常重要的,下面是來使用純javascript實(shí)現(xiàn)獲取服務(wù)器端的功能來展示如何使用純javascript實(shí)現(xiàn)ajax功能,以弄清其原理.2017-11-11