TypeScript中交叉類型和聯(lián)合類型的區(qū)別詳解
1. 定義
1.1. 聯(lián)合類型(|)
在TS中,聯(lián)合類型表示:一個(gè)值可以是多種類型之一,使用邏輯“或”( | )運(yùn)算符來(lái)分隔多個(gè)類型。
一個(gè)聯(lián)合類型的變量,在使用時(shí)可以是多個(gè)類型中的任意一種。
type UnionTypes = Type1 | Type2 | Type3;
1.2. 交叉類型(&)
在TS中,交叉類型表示:同時(shí)具備多種類型的值,使用邏輯“與”( & )運(yùn)算符進(jìn)行組合。
一個(gè)交叉類型的變量,將同時(shí)擁有多個(gè)類型的屬性和方法。
type IntersectionTypes = Type1 & Type2 & Type3;
2. 聯(lián)合類型
2.1. 基礎(chǔ)聯(lián)合類型
當(dāng)一個(gè)變量可以是多種不同的類型時(shí),可以使用聯(lián)合類型來(lái)定義它。
例如,一個(gè)變量可以是 string 類型或者 number 類型。
let data: string | number; data = 'hello ts'; data = 123; data = false; // 編譯錯(cuò)誤:不能將類型“boolean”分配給類型“string | number”。ts(2322)
上面這段代碼中,我們定義了一個(gè)變量 data,類型為 number 和 string 的聯(lián)合類型,因此,data 的值只能是這兩種類型中的其中一種,復(fù)制其它類型的值會(huì)報(bào)錯(cuò)。
2.2. 對(duì)象聯(lián)合類型
對(duì)象聯(lián)合類型只能訪問(wèn)聯(lián)合中所有類型共有的成員。
interface Admin { name: string; age: number; } interface User { name: string; sayHi(): void; } declare function Employee(): Admin | User; let employee = Employee(); employee.name = 'Echo'; // 下面語(yǔ)句會(huì)報(bào)錯(cuò):age屬性不是 Admin 和 User 共有的屬性 employee.age = 26; // 編譯錯(cuò)誤:類型“Admin | User”上不存在屬性“age”。類型“User”上不存在屬性“age”。ts(2339)
上面這段代碼中,定義了兩個(gè)接口 Admin 和 User,接著使用 declare function 聲明了一個(gè) Employee 函數(shù),該函數(shù)的返回類型為 Admin 或 User。之后通過(guò)調(diào)用 Employee 函數(shù)并將返回值賦給了 employee 變量。接著將 employee 對(duì)象中 name 屬性的值設(shè)置為 'Echo' 是可以的,因?yàn)?name 屬性是 Admin 和 User 共有的屬性。而將 employee 對(duì)象中 age 屬性的值設(shè)置為 26 時(shí)會(huì)出現(xiàn)編譯錯(cuò)誤,錯(cuò)誤信息指出類型 Admin | User 上不存在屬性 age。這是因?yàn)?age 屬性只存在于 Admin 接口中,而不屬于 User 接口。
造成該錯(cuò)誤的原因是,TypeScript 在聯(lián)合類型上只能訪問(wèn)聯(lián)合類型中所有類型的共有屬性和方法。 因此,通過(guò)聯(lián)合類型的變量只能訪問(wèn) name 屬性,而不能訪問(wèn) age 屬性。
2.3. 字面量聯(lián)合類型
聯(lián)合類型可以與字面量類型一起使用,用于限定一個(gè)值只能是某幾個(gè)特定的值之一。
let direction: "Up" | "Right" | "Down" | "Left"; direction = "Right"; direction = "none"; // 編譯錯(cuò)誤,只能取值為 "Up" | "Right" | "Down" | "Left"
3. 交叉類型
在 TypeScript 中,交叉類型(Intersection Types)允許我們將多個(gè)類型合并為一個(gè)新的類型。
使用交叉類型可以將多個(gè)對(duì)象的屬性和方法合并到一個(gè)新的對(duì)象中。
type Person = { name: string; } type User = { age: number; } let person: Person & User; person = { name: 'Echo', age: 26, } // 編譯錯(cuò)誤: // 不能將類型“{ name: string; }”分配給類型“Person & User”。 // 類型 "{ name: string; }" 中缺少屬性 "age",但類型 "User" 中需要該屬性。ts(2322) // index.ts(7, 3): 在此處聲明了 "age"。 person = { name: 'Steven', }
上面這段代碼中,我們定義了 Person 和 User 兩個(gè)類型,然后,我們定義一個(gè)變量 person,它的類型是使用交叉類型 Person & User 來(lái)創(chuàng)建的一個(gè)新類型,那么,此時(shí)變量 person 就同時(shí)具備了 name 和 age 屬性。
3.1. 交叉類型的成員類型是基礎(chǔ)類型
交叉類型的成員類型可以為任意類型,但需要注意的是,如果交叉類型的成員類型是基礎(chǔ)類型時(shí),交叉類型的結(jié)果是 never。
type T1 = string & number; // 等同于 type T1 = never type T2 = number & boolean; // 等同于 type T2 = never
3.2. 交叉類型的成員類型是對(duì)象類型
當(dāng)交叉類型的成員類型為對(duì)象類型時(shí),結(jié)果類型又會(huì)是什么?
下面我們看一個(gè)簡(jiǎn)單的例子:
type TypeA = { x: number; y: number; } type TypeB = { y: number; z: number; } type TypeC = { z: number; }
上面這段代碼中,我們定義了三個(gè)類型:TypeA、TypeB 和 TypeC,分別表示類型 A、B 和 C,類型 A 具有屬性成員 x 和 y,類型 B 具有屬性成員 y 和 z,類型 C 具有屬性成員 z,每個(gè)類型具有不同的屬性成員。
type MergedType = TypeA & TypeB & TypeC;
上面這段代碼中,我們使用交叉類型 TypeA & TypeB & TypeC 創(chuàng)建了一個(gè)新的類型 MergedType,它包含了類型 A、B 和 C 的屬性成員,那么,合并后的交叉類型的成員類型為:屬性成員 x 的類型是 A 的類型,屬性成員 y 的類型是 A 和 B 的交叉類型,屬性成員 z 的類型是 B 和 C 的交叉類型。
let t: MergedType; const t1 = { x: 1, y: 2, z: 3, } const t2 = { x: 10, y: 20, } t = t1; // 編譯錯(cuò)誤: // 不能將類型“{ x: number; y: number; }”分配給類型“MergedType”。 // 類型 "{ x: number; y: number; }" 中缺少屬性 "z",但類型 "TypeB" 中需要該屬性。ts(2322) // index.ts(10, 3): 在此處聲明了 "z"。 t = t2;
上面這段代碼中,定義了一個(gè)變量 t,它的類型是 TypeA & TypeB & TypeC 組成的交叉類型,然后再定義了兩個(gè)變量 t1 和 t2,t1 同時(shí)滿足 TypeA、TypeB 和 TypeC 類型約束,因此能賦值給交叉類型 t。而 t2 滿足 TypeA 類型約束,是 TypeA 類型,但并不能賦值給交叉類型 t,當(dāng) t2 賦值給 t 的時(shí)候,編譯器會(huì)報(bào)錯(cuò)。
由此可見(jiàn):交叉類型的類型成員由各個(gè)類型成員的屬性成員的并集組成,并且這些屬性成員的類型是各個(gè)成員類型的交叉類型。這種規(guī)則使得交叉類型能夠?qū)⒍鄠€(gè)類型的屬性成員合并到一個(gè)類型中,并且可以同時(shí)訪問(wèn)這些屬性成員。
3.3. 成員類型合并
如果交叉類型的成員類型中有相同的類型,合并后的交叉類型將只保留一份該成員的類型。
type T1 = string & string; // 等同于 type T1 = string type T2 = string & string & string; // 等同于 type T2 = string
上面這段代碼中,類型 T1 由兩個(gè) string 構(gòu)成,由于成員類型相同,所以合并成為一個(gè) string。類型 T2 由三個(gè) string 構(gòu)成,由于成員類型相同,所以合并成為一個(gè) string。
3.4. 交叉類型的索引簽名
當(dāng)交叉類型的成員類型之一具有數(shù)字索引簽名(即可通過(guò)數(shù)字索引訪問(wèn))或字符串索引簽名(即可通過(guò)字符串索引訪問(wèn))時(shí),結(jié)果類型也將包含相應(yīng)的數(shù)字索引簽名或字符串索引簽名。
結(jié)果類型的索引簽名值類型是各個(gè)成員類型索引簽名值類型的交叉類型。也就是說(shuō),通過(guò)交叉類型合并的結(jié)果類型的索引簽名值類型將是各個(gè)成員類型索引簽名值類型的交叉類型。
type TypeA = { [key: string]: string; }; type TypeB = { [key: number]: string; }; type MergedType = TypeA & TypeB; const mergedObject: MergedType = { name: 'Echo', gender: 'Male', city: 'Guang Zhou', 1: 'abcd', }; console.log(mergedObject['name']); // 輸出:Echo console.log(mergedObject['gender']); // 輸出:Male console.log(mergedObject['city']); // 輸出:Guang Zhou console.log(mergedObject[1]); // 輸出:abcd
上面這段代碼中,定義了兩個(gè)類型 TypeA 和 TypeB,其中,TypeA 具有字符串索引簽名,TypeB 具有數(shù)字索引簽名,也就是說(shuō),TypeA 允許使用字符串作為索引,而 TypeB 允許使用數(shù)字作為索引。然后,使用交叉類型 TypeA & TypeB 創(chuàng)建了一個(gè)新的類型 MergedType,它包含了 TypeA 和 TypeB 的索引簽名。接著,我們創(chuàng)建了一個(gè)名為 mergedObject 的對(duì)象,它的類型指定為交叉類型 MergedType,該對(duì)象可以通過(guò)數(shù)字索引或字符串索引來(lái)訪問(wèn),并給這些索引賦予了相應(yīng)的值。最后,我們通過(guò)索引訪問(wèn) mergedObject 對(duì)象的值來(lái)驗(yàn)證交叉類型的索引簽名的合并情況。
3.5. 交叉類型的調(diào)用簽名
當(dāng)交叉類型的成員類型中至少有一個(gè)具有調(diào)用簽名時(shí),交叉類型的結(jié)果類型也會(huì)包含這個(gè)調(diào)用簽名。
換句話說(shuō),交叉類型中至少一個(gè)成員的調(diào)用簽名會(huì)被合并到結(jié)果類型中。
此外,如果交叉類型的多個(gè)成員類型都有調(diào)用簽名,那么結(jié)果類型將會(huì)形成調(diào)用簽名重載的結(jié)構(gòu)。調(diào)用簽名重載允許我們?yōu)橥粋€(gè)函數(shù)提供多個(gè)不同的調(diào)用方式,具體取決于參數(shù)類型和返回值類型。
可以將交叉類型的成員類型的調(diào)用簽名視為函數(shù)的簽名,交叉類型的結(jié)果類型即為這些簽名的合并。
type FunctionA = (x: number, y: number) => number; type FunctionB = (x: string, y: string) => string; type FunctionType = FunctionA & FunctionB; const option: FunctionType = (x, y) => x + y; console.log(option(10, 20)); // 輸出: 30 console.log(option('a', 'b')); // 輸出: ab
上面這段代碼中,定義了兩個(gè)類型 FunctionA 和 FunctionB,它們接收 x 和 y 兩個(gè)參數(shù),其中,F(xiàn)unctionA 兩個(gè)參數(shù)的類型和函數(shù)返回值的類型都是 number 類型,F(xiàn)unctionB 兩個(gè)參數(shù)的類型和函數(shù)返回值的類型都是 string 類型。然后,使用交叉類型 FunctionA & FunctionB 創(chuàng)建了一個(gè)新的類型 FunctionType,這個(gè)交叉類型包含了兩個(gè)成員類型的調(diào)用簽名。最后,我們創(chuàng)建了一個(gè)名為 option 的變量,它的類型被定義為 FunctionType,也就是 FunctionA 和 FunctionB 的交叉類型。我們可以使用 option 等同于調(diào)用兩個(gè)函數(shù)的方式來(lái)執(zhí)行相應(yīng)的運(yùn)算,option(10, 20) 相當(dāng)于加法運(yùn)算,輸出結(jié)果為:30,option('a', 'b') 相當(dāng)于字符串的拼接,輸出結(jié)果為:ab。
3.6. 交叉類型的構(gòu)造簽名
當(dāng)交叉類型的成員類型中至少有一個(gè)具有構(gòu)造簽名時(shí),交叉類型的結(jié)果類型也會(huì)包含這個(gè)構(gòu)造簽名。
換句話說(shuō),交叉類型中至少存在一個(gè)成員的構(gòu)造簽名會(huì)被合并到結(jié)果類型中。
如果交叉類型的多個(gè)成員類型都具有構(gòu)造簽名,那么結(jié)果類型將形成構(gòu)造簽名重載的結(jié)構(gòu)。構(gòu)造簽名重載允許我們?yōu)橥粋€(gè)類提供多個(gè)不同的構(gòu)造方式,具體取決于參數(shù)列表。
interface Foo { new (name: string): string } interface Bar { new (name: number): number; } type FooBar = Foo & Bar; declare const T: FooBar; const instance1 = new T('Echo'); const instance2 = new T(26);
上面這段代碼中,我們定義了兩個(gè)接口 Foo 和 Bar,它們都具有構(gòu)造簽名,分別接受不同的參數(shù)類型并返回對(duì)應(yīng)的類型。接著,我們使用交叉類型 Foo & Bar 創(chuàng)建了一個(gè)新的類型 FooBar,它是 Foo 和 Bar 的交叉類型,意味著 FooBar 同時(shí)具備了 Foo 和 Bar 接口的構(gòu)造簽名。然后,通過(guò) declare 關(guān)鍵字聲明了一個(gè)常量 T,它的類型被定義為 FooBar。接著我們創(chuàng)建了兩個(gè)實(shí)例 instance1 和 instance2。由于 T 的類型為 FooBar,我們可以使用 new T 的語(yǔ)法來(lái)實(shí)例化對(duì)象。對(duì)于 instance1,使用字符串 'Echo' 作為參數(shù)傳遞給構(gòu)造函數(shù),這符合 Foo 接口中定義的構(gòu)造簽名,所以實(shí)例化成功,返回一個(gè)字符串類型的實(shí)例。對(duì)于 instance2,使用數(shù)字 26 作為參數(shù)傳遞給構(gòu)造函數(shù),這符合 Bar 接口中定義的構(gòu)造簽名,所以實(shí)例化成功,返回一個(gè)數(shù)字類型的實(shí)例。
4. 總結(jié)
- 聯(lián)合類型只能訪問(wèn)共有的屬性和方法。例如,如果一個(gè)變量是 number 類型或者 string 類型,那么只能使用這兩種類型共有的屬性和方法。
- 聯(lián)合類型中的變量,如果在特定條件下可以判斷出其具體的類型,可以使用類型斷言(as語(yǔ)法)來(lái)告訴編譯器具體的類型。
- 交叉類型的結(jié)果包含了所有成員類型的屬性和方法,通過(guò)合并同名成員實(shí)現(xiàn)。屬性會(huì)合并為并集類型,方法會(huì)合并為聯(lián)合類型。
- 當(dāng)成員類型具有調(diào)用簽名或構(gòu)造簽名時(shí),交叉類型的結(jié)果將形成相應(yīng)簽名的重載。
以上就是TypeScript中交叉類型和聯(lián)合類型的區(qū)別詳解的詳細(xì)內(nèi)容,更多關(guān)于TypeScript交叉類型和聯(lián)合類型區(qū)別的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
利用原生JS自動(dòng)生成文章標(biāo)題樹的實(shí)例
網(wǎng)上關(guān)于生成文章標(biāo)題樹的示例很多,這篇文章介紹的是利用原生JS實(shí)現(xiàn)自動(dòng)生成文章標(biāo)題樹,實(shí)現(xiàn)過(guò)程很簡(jiǎn)單,有需要的可以參考借鑒。2016-08-08Angularjs結(jié)合Bootstrap制作的一個(gè)TODO List
這篇文章主要介紹了Angularjs結(jié)合Bootstrap制作的一個(gè)TODO List 的相關(guān)資料,感興趣的小伙伴們可以參考一下2016-08-08web前端開發(fā)中常見(jiàn)的多列布局解決方案整理(一定要看)
多列布局在web前端開發(fā)中也是較為常見(jiàn)的,今天小編給大家介紹這里會(huì)提到的多列布局有兩列定寬加一列自適應(yīng)、多列不定寬加一列自適應(yīng)、多列等分三種,感興趣的朋友一起看看吧2017-10-10javascript showModalDialog 內(nèi)跳轉(zhuǎn)頁(yè)面的問(wèn)題
在頁(yè)面中使用了showModalDialog,但是在跳轉(zhuǎn)鏈接時(shí),不會(huì)在當(dāng)前頁(yè)執(zhí)行,而是彈出一個(gè)新的頁(yè)面。2010-11-11JS動(dòng)態(tài)遍歷json中所有鍵值對(duì)的方法(不知道屬性名的情況)
這篇文章主要介紹了JS動(dòng)態(tài)遍歷json中所有鍵值對(duì)的方法,實(shí)例分析了針對(duì)不知道屬性名的情況簡(jiǎn)單遍歷json鍵值對(duì)的操作技巧,需要的朋友可以參考下2016-12-12微信小程序?qū)崿F(xiàn)發(fā)動(dòng)態(tài)功能的示例代碼
最近做了一個(gè)校園拍賣小程序,想在里面添加一個(gè)類似校園圈功能,現(xiàn)在來(lái)一步一步實(shí)現(xiàn),對(duì)微信小程序?qū)崿F(xiàn)發(fā)動(dòng)態(tài)功能感興趣的朋友一起看看吧2022-08-08JavaScrip報(bào)錯(cuò):module?is?not?defined的原因及解決
這篇文章主要給大家介紹了關(guān)于JavaScrip報(bào)錯(cuò):module?is?not?defined的原因及解決方法,文中通過(guò)代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-09-09