詳解TypeScript中的箭頭函數(shù)如何實(shí)現(xiàn)重載
這個(gè)問(wèn)題來(lái)自于我在網(wǎng)上搜索的時(shí)候,基本上清一色的翻譯官網(wǎng)的函數(shù)重載的章節(jié)的內(nèi)容,對(duì)于我想要的箭頭函數(shù)的重載沒(méi)有太多幫助。包括官網(wǎng),其實(shí)也沒(méi)有非常明確的說(shuō)明箭頭函數(shù)該如何重載。
這里先直接上結(jié)論,在 ts 中,可以借助 Call Signatures 這個(gè)特性來(lái)實(shí)現(xiàn)箭頭函數(shù)的重載。原本是用來(lái)給函數(shù)聲明增加靜態(tài)屬性的,但是卻可以用來(lái)完成箭頭函數(shù)的類(lèi)型聲明。
type Test = { (s: string): string; (s: number): number; (s: string, b: number): number; }; type getState<T> = { (): T; <K extends keyof T>(key: K): T[K]; };
本質(zhì)就是定一個(gè)新的類(lèi)型,鍵值就用括號(hào)包,里面就是不同的入?yún)?,函?shù)的返回值就是鍵值即可。
但是不出意外的話,意外就會(huì)發(fā)生,在實(shí)操的時(shí)候,往往這么寫(xiě)非常不 ok。
實(shí)操
實(shí)戰(zhàn)中這么寫(xiě),類(lèi)型聲明是好寫(xiě)的,但是函數(shù)的實(shí)現(xiàn),其實(shí)并不好寫(xiě),以這樣一個(gè)函數(shù)為例:
const deal = (a: number | string, b: number | string): number | string => { if (typeof a === 'number' && typeof b === 'number') { return a + b; } if (typeof a === 'string' || typeof b === 'string') { return String(a) + String(b); } return 0; };
當(dāng)入?yún)⒍际?number 時(shí),返回 number,入?yún)⒂幸粋€(gè)是 string 時(shí),返回 string。
那么對(duì)于這樣一個(gè)函數(shù),其實(shí)存在幾個(gè)重載的情況:
type Deal1 = { (a: number, b: number): number; (a: number, b: string): string; (a: string, b: number): string; (a: string, b: string): string; };
要想把上述類(lèi)型賦值給 deal 函數(shù),會(huì)出現(xiàn)返回值匹配不上的問(wèn)題。
// ts 會(huì)提示類(lèi)型錯(cuò)誤: // Type '(a: number | string, b: number | string) => number | string' is not assignable to type 'Deal1'. // Type 'string | number' is not assignable to type 'number'. const deal: Deal1 = ( a: number | string, b: number | string, ): number | string => { if (typeof a === 'number' && typeof b === 'number') { return a + b; } if (typeof a === 'string' || typeof b === 'string') { return String(a) + String(b); } ??????? return 0; };
從現(xiàn)象來(lái)看,ts 對(duì)于這樣的寫(xiě)法在做檢查的時(shí)候,會(huì)將當(dāng)前函數(shù)對(duì)重載的幾個(gè)類(lèi)型都進(jìn)行檢查,看看類(lèi)型上是否能夠賦值,相當(dāng)于:
// 偽代碼,理解意思就行 check1: (a: number, b: number) => number = (a: number | string, b: number | string) => number | string); check2: (a: number, b: string) => string = (a: number | string, b: number | string) => number | string); check3: (a: string, b: number) => string = (a: number | string, b: number | string) => number | string); check4: (a: string, b: string) => string = (a: number | string, b: number | string) => number | string);
對(duì)于入?yún)ⅲ捎谑悄孀兾恢?,所?number = number | string 能夠賦值,所以參數(shù)的類(lèi)型能夠通過(guò)校驗(yàn),而返回值屬于順變位置,所以 number = number | string 是不能通過(guò)類(lèi)型校驗(yàn)的。
想要將返回值賦值成功,返回值必須是 number & string 或者 any,前者就是 never 了,此時(shí)會(huì)發(fā)現(xiàn)雖然賦值通過(guò)了 deal 的校驗(yàn),但是函數(shù)的實(shí)現(xiàn)中,就會(huì)報(bào)返回值錯(cuò)誤的問(wèn)題。如果改成 any,那么雖然不會(huì)報(bào)錯(cuò),但是在函數(shù)中就缺失了對(duì)返回值的檢查。如下:
// 返回值改為 number & string, // 賦值處能夠避免類(lèi)型錯(cuò)誤 const deal: Deal1 = ( a: number | string, b: number | string, ): number & string => { if (typeof a === 'number' && typeof b === 'number') { // 此時(shí)返回值是 never,此處會(huì)報(bào)類(lèi)型錯(cuò)誤 return a + b; } if (typeof a === 'string' || typeof b === 'string') { // 此時(shí)返回值是 never,此處會(huì)報(bào)類(lèi)型錯(cuò)誤 return String(a) + String(b); } // 此時(shí)返回值是 never,此處會(huì)報(bào)類(lèi)型錯(cuò)誤 return 0; }; // 返回值改為 any, // 賦值處能夠避免類(lèi)型錯(cuò)誤 const deal: Deal1 = (a: number | string, b: number | string): any => { if (typeof a === 'number' && typeof b === 'number') { return a + b; } if (typeof a === 'string' || typeof b === 'string') { return String(a) + String(b); } // 此處寫(xiě)任何類(lèi)型都不會(huì)拋錯(cuò),缺失了原本的期望的校驗(yàn) return undefined; };
那么目前看,想要用這種方式實(shí)現(xiàn)箭頭函數(shù)的重載,就只能將返回值設(shè)定為 any,這樣,雖然在用戶使用的時(shí)候能夠進(jìn)行非常好的類(lèi)型提示,但是開(kāi)發(fā)者本身不能再借助 ts 完成對(duì)這個(gè)函數(shù)的返回值的校驗(yàn)。
此時(shí)還有一種寫(xiě)法,就是 as,寫(xiě)法如下:
const deal = ((a: string | number, b: number | string): number | string => { if (typeof a === 'number' && typeof b === 'number') { return a + b; } if (typeof a === 'string' || typeof b === 'string') { return String(a) + String(b); } return 0; }) as Deal1;
這樣的寫(xiě)法,即能夠滿足函數(shù)本身的返回值校驗(yàn) (可以把 0 改成其他類(lèi)型試試),同時(shí),又具備了 Deal1 的重載的類(lèi)型聲明:
// case1: number const case1 = deal(1, 2); // case2: string const case2 = deal('1', 2); // case3: never,入?yún)㈩?lèi)型錯(cuò)誤 const case3 = deal({}, 2);
最佳實(shí)踐
附上完整的代碼:
// 重載類(lèi)型聲明 type Deal1 = { (a: number, b: number): number; (a: number, b: string): string; (a: string, b: number): string; (a: string, b: string): string; }; const deal = (( // 此時(shí)入?yún)ⅲ祷刂档念?lèi)型可自行限制 // 無(wú)需掛念 Deal1 中的定義 a: string | number, b: number | string, ): number | string => { if (typeof a === 'number' && typeof b === 'number') { return a + b; } if (typeof a === 'string' || typeof b === 'string') { return String(a) + String(b); } // 這樣寫(xiě),原函數(shù)具備校驗(yàn)的能力 return 0; // 通過(guò) as 指定類(lèi)型 }) as Deal1; const case1 = deal(1, 2); const case2 = deal('1', 2); const case3 = deal({}, 2);
核心就是原函數(shù)類(lèi)型寫(xiě)法一致,重載的類(lèi)型通過(guò) as 進(jìn)行賦值,這樣就兼顧了類(lèi)型提示和原函數(shù)的類(lèi)型校驗(yàn)。
其他方法
由于 type 和 interface 的用法在此處并無(wú)歧義,所以換成 interface 也是 ok。
另一種就是函數(shù)的交叉,也是實(shí)現(xiàn)重載的一種方案,如下:
type Deal3 = ((a: number, b: number) => number) & ((a: number, b: string) => string) & ((a: string, b: number) => string) & ((a: string, b: string) => string);
總結(jié)
本文介紹了 TS 實(shí)現(xiàn)箭頭函數(shù)重載的幾種方案,借助了 Call Signatures 的特性。
同時(shí)給出了這樣寫(xiě),在實(shí)戰(zhàn)中可能會(huì)遇到的類(lèi)型不匹配的問(wèn)題及解決方案。希望能夠?qū)τ龅酵瑯訂?wèn)題的同學(xué)有所幫助吧。
到此這篇關(guān)于詳解TypeScript中的箭頭函數(shù)如何實(shí)現(xiàn)重載的文章就介紹到這了,更多相關(guān)TypeScript函數(shù)重載內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
window.onload 加載完畢的問(wèn)題及解決方案(上)
我們經(jīng)常使用 window.onload 來(lái)處理頁(yè)面,當(dāng)頁(yè)面加載完成做一些事情。但這個(gè) window.onload 是頁(yè)面全部加載完成,甚至包括圖片,而我們實(shí)際上經(jīng)常需要的是文檔 DOM 加載完畢!2009-07-07JavaScript實(shí)現(xiàn)淺拷貝與深拷貝的方法分析
這篇文章主要介紹了JavaScript實(shí)現(xiàn)淺拷貝與深拷貝的方法,結(jié)合實(shí)例形式總結(jié)分析了JavaScript淺拷貝與深拷貝的定義與使用方法,需要的朋友可以參考下2018-07-07JS實(shí)現(xiàn)拖動(dòng)模態(tài)框案例
這篇文章主要為大家詳細(xì)介紹了JS實(shí)現(xiàn)拖動(dòng)模態(tài)框案例,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-07-07ES6中l(wèi)et、const的區(qū)別及變量的解構(gòu)賦值操作方法實(shí)例分析
這篇文章主要介紹了ES6中l(wèi)et、const的區(qū)別及變量的解構(gòu)賦值操作方法,結(jié)合實(shí)例形式分析了ES6中l(wèi)et、const的功能、原理、使用方法及數(shù)組、字符串、函數(shù)參數(shù)等解構(gòu)賦值相關(guān)操作技巧,需要的朋友可以參考下2019-10-10用JavaScript實(shí)現(xiàn)簡(jiǎn)單網(wǎng)頁(yè)時(shí)鐘
這篇文章主要為大家詳細(xì)介紹了用JavaScript實(shí)現(xiàn)簡(jiǎn)單網(wǎng)頁(yè)時(shí)鐘,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08js實(shí)時(shí)獲取窗口大小變化的實(shí)例代碼
下面小編就為大家?guī)?lái)一篇js實(shí)時(shí)獲取窗口大小變化的實(shí)例代碼。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-11-11JavaScript?排他思想的具體實(shí)現(xiàn)
排他思想的算法就是排除掉其他的,本文主要介紹了JavaScript?排他思想的實(shí)現(xiàn),以及介紹了兩個(gè)示例,感興趣的可以了解一下2021-11-11微信小程序仿微信運(yùn)動(dòng)步數(shù)排行(交互)
這篇文章主要介紹了微信小程序仿微信運(yùn)動(dòng)步數(shù)排行(交互),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-07-07使用Cookies保存網(wǎng)站歷史瀏覽記錄實(shí)例代碼
仿淘寶網(wǎng)的最近瀏覽記錄功能,喜歡的朋友可以參考下。2010-07-07