如何在TypeScript中使用函數(shù)
前言
雖然 JS/TS 支持面向?qū)ο缶幊?,但大部分時(shí)候還是在寫函數(shù)。函數(shù)是一等公民。本文介紹下如何在 TypeScript 中使用函數(shù),包括:
- 函數(shù)類型聲明
- 函數(shù)參數(shù)類型:可選參數(shù)、默認(rèn)參數(shù)、剩余參數(shù)
- 函數(shù)返回值類型
- this 類型
- 函數(shù)重載
函數(shù)類型
面試中經(jīng)常會被問到,JS 中有哪幾種數(shù)據(jù)類型。其中就會有函數(shù)類型。
JS 中的函數(shù)類型很模糊,準(zhǔn)確來說,僅有類型的概念,卻無類型的實(shí)質(zhì)。好在有了 TS 強(qiáng)類型的加持,現(xiàn)在可以好好定義一個(gè)函數(shù)的類型了。
聲明一個(gè)函數(shù)類型,有若干種不同的方式。比如通過 interface 定義函數(shù)類型,通過 type 聲明函數(shù)類型別名,在類聲明中聲明函數(shù)類型等等。
但核心就一點(diǎn),只關(guān)注函數(shù)參數(shù)的類型和返回值的類型。
下面就從函數(shù)的聲明入手,來看看函數(shù)類型的使用。
函數(shù)的聲明方式
有三種最簡單的函數(shù)聲明方式。
使用 function
關(guān)鍵字進(jìn)行聲明:
function add(a: number, b: number): number { return a + b }
通過函數(shù)表達(dá)式進(jìn)行聲明:
let add = function (a: number, b: number): number { return a + b }
或者使用箭頭函數(shù):
let add = (a: number, b: number): number => { return a + b }
上面這三種聲明函數(shù)的方式和 JS 中聲明函數(shù)別無二致,就是多了兩點(diǎn):
- 函數(shù)參數(shù)需要給明類型
- 函數(shù)返回值也需要給出類型
那么函數(shù)類型到底在哪里呢?把鼠標(biāo)移動(dòng)到函數(shù)名字上就能看到了:
和聲明普通變量變量一樣:
let name = 'kunwu'
如果沒有使用類型注解,那么編譯器會自動(dòng)推導(dǎo)出類型來。上面紅框標(biāo)識出來的類型,就是編譯器推導(dǎo)出來的。
函數(shù)的類型注解
函數(shù)的類型形如 (參數(shù):類型) => 類型,所以函數(shù)的類型注解就是:
let add: (a: number, b: number) => number = function (a, b) { return a + b }
通常函數(shù)的類型都比較長,此時(shí)可以使用 type 類型別名,來聲明一個(gè)標(biāo)識符表示該類型:
type Add = (a: number, b: number) => number let add: Add => number = function (a, b) { return a + b }
這樣,聲明函數(shù)類型,使用函數(shù)類型,看上去就很簡潔直觀了。
對象中的函數(shù)類型
函數(shù)的參數(shù)
聲明函數(shù)時(shí),可以使用類型注解聲明參數(shù)類型。如果不指定類型,默認(rèn)類型是 any
。
function add (a: number, b: number) :number { return a + b }
可選參數(shù)
可選參數(shù)使用 ?
標(biāo)記,需要放到固定參數(shù)的后面:
const fn = (a: number, b?:number) => { if(b) { return a + b } else { return a } }
可選參數(shù)意味著參數(shù)可傳可不傳。通過編譯器的類型推導(dǎo),可以看到好像可選參數(shù)等同于一個(gè)聯(lián)合類型:
但其實(shí)并不等同。
可選參數(shù)表示參數(shù)可傳可不傳,若傳則必須是指定的類型。
而直接將參數(shù)類型聲明為 string|undefined
,意味著這是個(gè)必傳參數(shù),必須傳實(shí)參。
參數(shù)默認(rèn)值
參數(shù)的默認(rèn)值,在參數(shù)的類型后面使用 =
聲明:
const fn = (a: number, b: number = 10): number => { return a + b }
剩余參數(shù)
剩余參數(shù)是 ES6 中的一個(gè)特性,使用剩余運(yùn)算符 ...
來獲取函數(shù)實(shí)參中未被形參聲明的變量,其取得的值是一個(gè)數(shù)組,比如:
function add(a,b, ...args) { console.log(args) } add(1,2,3,4,5)
則剩余參數(shù) args 就等于 [3, 4, 5]。
TS 中剩余參數(shù)也要有類型聲明,需要將其聲明為數(shù)組類型或者元組類型。
function add(a: number, b: number, ...args: number[]) { console.log(c) }
剩余參數(shù)和其他的固定參數(shù)不同。其他的固定參數(shù)聲明了類型,則必須傳該類型的值。而剩余參數(shù)雖然也聲明了類型,但可傳可不傳,可傳一個(gè),可傳多個(gè),比如:
function add(a: number, b: number, ...args: number[]) { console.log(args) } add(1, 2) // [ ] add(1, 2, 3) // [ 3 ] add(1, 2, 3, 4) // [ 3, 4 ]
還可以將剩余參數(shù)類型聲明為元組類型,元組中還可以繼續(xù)使用可選參數(shù),在參數(shù)后使用 ?
表示:
function log( ...args: [string, number, boolean?]) { console.log(args) } log('Shinichi', 17) // [ 'Shinichi', 17 ] log('Zoro', 19, true) // [ 'Zoro', 19, true ]
函數(shù)的返回值類型
函數(shù)的返回值類型可以通過類型注解指定。如果不指定的話,TS 編譯器能夠根據(jù)函數(shù)體的 return 語句自動(dòng)推斷出返回值類型,因此我們也可以省略返回值類型。
TS 基本類型中有一個(gè) void 類型,表示空類型,它唯一的用處就是用作函數(shù)的返回值類型。當(dāng)一個(gè)函數(shù)沒有 return 語句時(shí),
如果函數(shù)使用 return 語句返回了 undefined 值,則返回值類型就為 undefined 而不是 void 了:
但是此時(shí)可以將返回值類型使用類型注解指定為 void:
this 類型
JS 函數(shù)中到處可見 this 的身影。關(guān)于 this 的指向也是前端八股中的基礎(chǔ)之基礎(chǔ)。
默認(rèn)情況下 TS 編譯器會將函數(shù)中的 this 設(shè)為 any
類型,也就是說編譯器不會對 this 做類型檢查,就可以任意使用 this,比如:
function fn() { this.name = 'Naruto' this.age = 18 }
同時(shí) TS 編譯器又提供了一個(gè)編譯選項(xiàng) --noImplicitThis
,開啟后會檢查 this 的類型,此時(shí)需要明確指定其類型,否則會報(bào)錯(cuò),如下:
那么如何為 this 聲明類型呢?
聲明 this 類型
TS 函數(shù)中有一個(gè)特殊的 this 參數(shù),它用來指定該函數(shù)中用到的 this 的類型,需要定義在形參的第一個(gè)位置。
還是上面的例子,要為函數(shù)中的 this 指定類型的話,這樣寫:
function fn(this: {name: string, age: number}) { this.name = 'Naruto' this.xxx = 'xxx' }
直接在函數(shù)參數(shù)列表中聲明 this 類型不太優(yōu)雅,可以使用 type 關(guān)鍵字聲明別名再使用:
type Person = {name: string, age: number} function fn(this: Person) { this.name = 'Naruto' this.age = 18 }
當(dāng)定義了 this 類型后,函數(shù)在調(diào)用時(shí)也會有所不同,需要使用 call、apply :
type Person = {name: string, age: number} function fn(this: Person, a: number) { this.name = 'Naruto' this.age = 18 } fn.call({name: 'Naruto', age: 18}, 10) fn.apply({name: 'Naruto', age: 18}, [10])
像以前那樣直接調(diào)用函數(shù)是錯(cuò)誤的:
fn(10) // X
函數(shù)重載
面向?qū)ο缶幊逃腥筇卣?,封裝、繼承和多態(tài)。多態(tài)的表現(xiàn)之一就是函數(shù)的重載。
函數(shù)重載,就是可以多次聲明一個(gè)同名函數(shù),但是它們的參數(shù)類型不同或者參數(shù)個(gè)數(shù)不同。這樣在調(diào)用時(shí),可以根據(jù)傳入?yún)?shù)類型的不同,參數(shù)個(gè)數(shù)的不同,來確定執(zhí)行的到底是哪一個(gè)函數(shù)。
函數(shù)重載是后端語言的概念,比如 java 中:
// 兩個(gè)整數(shù)的和 public static int add(int a, int b) { return a+b; } // 三個(gè)整數(shù)的和 public static int add(int a, int b, int c) { return add(a,b)+c; }
JS 本身并不支持函數(shù)重載。如果多次聲明一個(gè)函數(shù),則后聲明的會覆蓋掉先聲明的。但是 JS 可以利用 arguments 對象,再加上判斷實(shí)參的類型,來模擬重載的功能,比如:
function add() { let args = Array.from(arguments) if(args.length == 2) { return args[0] + args[1] } else if(args.length == 3) { return args[0] + args[1] + args[3] } } add(1,2) add(1, 2, 3)
TS 中多次聲明一個(gè)同名函數(shù),編譯器會報(bào)錯(cuò)。但是 TS 提供了實(shí)現(xiàn)函數(shù)重載的方法:先通過 function 關(guān)鍵字聲明重載的類型,最后寫函數(shù)的實(shí)現(xiàn)。
function add(a: number, b: number) : number function add(a: string, b: string) : string function add(a: number|string, b: number | string) { if(typeof a === 'number' && typeof b === 'number') { return a + b } else if(typeof a === 'string' && typeof b === 'string') { return a + b } } add(1, 2) add('10', '20')
由于在聲明重載時(shí)已經(jīng)確定了函數(shù)的返回值類型,在寫函數(shù)實(shí)現(xiàn)時(shí),就不再需要寫返回值類型了。編譯器會根據(jù)重載類型自動(dòng)推導(dǎo)。
TS 的函數(shù)重載只是一種偽重載,最終還是要靠去判斷類型,來執(zhí)行不同的邏輯。
還有一點(diǎn)要注意,聲明函數(shù)重載和函數(shù)實(shí)現(xiàn)必須寫在一塊,中間不能插入其他語句。
總結(jié)
本文介紹了TS 中有關(guān)函數(shù)的知識,包括函數(shù)的聲明方式,如何聲明函數(shù)類型,函數(shù)參數(shù)和返回值的類型,函數(shù)重載以及 this 的類型。大部分內(nèi)容和 JS 中差不太多,主要是 this 類型和函數(shù)重載這兩點(diǎn),需要額外關(guān)注下。
以上就是如何在TypeScript中使用函數(shù)的詳細(xì)內(nèi)容,更多關(guān)于TypeScript 函數(shù)使用的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
JavaScript實(shí)現(xiàn)簡單隨機(jī)點(diǎn)名器
這篇文章主要為大家詳細(xì)介紹了JavaScript實(shí)現(xiàn)簡單隨機(jī)點(diǎn)名器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11d3.js實(shí)現(xiàn)簡單的網(wǎng)絡(luò)拓?fù)鋱D實(shí)例代碼
最近一直在學(xué)習(xí)d3.js,大家都知道d3.js是一個(gè)非常不錯(cuò)的數(shù)據(jù)可視化庫,我們可以用它來做一些比較酷的東西,比如可以來顯示一些簡單的網(wǎng)絡(luò)拓?fù)鋱D,這篇文中就通過實(shí)例代碼給大家介紹了如何利用d3.js實(shí)現(xiàn)簡單的網(wǎng)絡(luò)拓?fù)鋱D,有需要的朋友們可以參考借鑒,下面來一起看看吧。2016-11-11淺談js之字面量、對象字面量的訪問、關(guān)鍵字in的用法
下面小編就為大家?guī)硪黄獪\談js之字面量、對象字面量的訪問、關(guān)鍵字in的用法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-11-11JavaScript編寫Chrome擴(kuò)展實(shí)現(xiàn)與瀏覽器的交互及時(shí)間通知
得益于API,我們可以用JavaScript編寫Chrome擴(kuò)展實(shí)現(xiàn)與瀏覽器的交互及時(shí)間通知。值得一提的是現(xiàn)在Chrome擁有后臺進(jìn)程可以使通知在前臺瀏覽器關(guān)閉的情況下依然能夠生效.2016-05-05IE6,IE7,IE8下使用Javascript記錄光標(biāo)選中范圍(已補(bǔ)全)
IE6,7,8下使用Javascript記錄光標(biāo)選中范圍(已補(bǔ)全)(已解決單個(gè)節(jié)點(diǎn)內(nèi)部重復(fù)字符的問題)2011-08-08