函数
函数是 JavaScript 应用程序的基础。 它帮助实现抽象层,模拟类,信息隐藏和模块。 在TypeScript里,虽然已经支持类,命名空间和模块,但函数仍然是主要的定义行为的地方。 TypeScript 为 JavaScript 函数添加了额外的功能,可以更容易地使用
基本示例
和 JavaScript 一样,TypeScript 函数可以创建有名字的函数和匿名函数。 可以随意选择适合应用程序的方式,不论是定义一系列API函数还是只使用一次的函数
通过下面的例子可以迅速回想起这两种 JavaScript 中的函数:
// 命名函数
function add(x, y) {
return x + y
}
// 匿名函数
let myAdd = function (x, y) {
return x + y
}
在JavaScript里,函数可以使用函数体外部的变量。当函数这么做时,我们说它‘捕获’了这些变量。 至于为什么可以这样做以及其中的利弊超出了本文的范围,但是深刻理解这个机制对学习 JavaScript 和 TypeScript 会很有帮助
let z = 100
function addToZ(x, y) {
return x + y + z
}
函数类型
为函数定义类型
为上面那个函数添加类型:
function add(x: number, y: number): number {
return x + y
}
let myAdd = function (x: number, y: number): number {
return x + y
}
可以给每个参数添加类型之后再为函数本身添加返回值类型。 TypeScript 能够根据返回语句自动推断出返回值类型,因此通常省略它
书写完整函数类型
现在已经为函数指定了类型,下面写出函数的完整类型
let myAdd: (x: number, y: number) => number = function (
x: number,
y: number
): number {
return x + y
}
函数类型包含两部分:参数类型和返回值类型。 当写出完整函数类型的时候,这两部分都是需要的。 以参数列表的形式写出参数类型,为每个参数指定一个名字和类型。 这个名字只是为了增加可读性。 也可以这么写:
let myAdd: (baseValue: number, increment: number) => number = function (
x: number,
y: number
): number {
return x + y
}
只要参数类型是匹配的,那么就认为它是有效的函数类型,而不在乎参数名是否正确
第二部分是返回值类型。 对于返回值,在函数和返回值类型之前使用(=>)符号,使之清晰明了。 如之前提到的,返回值类型是函数类型的必要部分,如果函数没有返回任何值,也必须指定返回值类型为 void
而不能留空
函数的类型只是由参数类型和返回值类型组成的。 函数中使用的捕获变量不会体现在类型里。 实际上,这些变量是函数的隐藏状态并不是组成API的一部分
推断类型
尝试这个例子的时候,会注意到,就算仅在等式的一侧带有类型,TypeScript编译器仍可正确识别类型:
let myAdd = function (x: number, y: number): number {
return x + y
}
let myAdd: (baseValue: number, increment: number) => number = function (x, y) {
return x + y
}
这叫做“按上下文归类”,是类型推论的一种。 可以更好地为程序指定类型
可选参数和默认参数
TypeScript 里的每个函数参数都是必须的。 这不是指不能传递 null
或 undefined
作为参数,而是说编译器检查用户是否为每个参数都传入了值。 编译器还会假设只有这些参数会被传递进函数。 简短地说,传递给一个函数的参数个数必须与函数期望的参数个数一致
function buildName(firstName: string, lastName: string) {
return firstName + ' ' + lastName
}
let result1 = buildName('Bob') // Error, 参数过少
let result2 = buildName('Bob', 'Adams', 'Sr.') // Error, 参数过多
let result3 = buildName('Bob', 'Adams') // OK
JavaScript 里,每个参数都是可选的,可传可不传。 没传参的时候,它的值就是 undefined
。 在 TypeScript 里可以在参数名旁使用 ?
实现可选参数的功能。 比如,让 lastName
是可选的:
function buildName(firstName: string, lastName?: string) {
if (lastName) return firstName + ' ' + lastName
else return firstName
}
let result1 = buildName('Bob') // OK
let result2 = buildName('Bob', 'Adams', 'Sr.') // Error, 参数过多
let result3 = buildName('Bob', 'Adams') // OK
可选参数必须跟在必须参数后面。 如果上例想让 firstName
是可选的,那么就必须调整它们的位置,把 firstName
放在后面
在 TypeScript 里,也可以为参数提供一个默认值(当用户没有传递这个参数或传递的值是 undefined
时)。它们叫做有默认初始化值的参数。 修改上例,把 lastName
的默认值设置为 "Smith"
function buildName(firstName: string, lastName = 'Smith') {
return firstName + ' ' + lastName
}
let result1 = buildName('Bob') // Bob Smith
let result2 = buildName('Bob', undefined) // Bob Smith
let result3 = buildName('Bob', 'Adams', 'Sr.') // 错误, 参数过多
let result4 = buildName('Bob', 'Adams') // Bob Adams
与普通可选参数不同的是,带默认值的参数不需要放在必须参数的后面。 如果带默认值的参数出现在必须参数前面,用户必须明确的传入undefined
值来获得默认值。 例如,重写最后一个例子,让 firstName
是带默认值的参数:
function buildName(firstName = 'Will', lastName: string) {
return firstName + ' ' + lastName
}
let result1 = buildName('Bob') // Error, 参数过少
let result2 = buildName('Bob', 'Adams', 'Sr.') // Error, 参数过多
let result3 = buildName('Bob', 'Adams') // Bob Adams
let result4 = buildName(undefined, 'Adams') // Will Adams
剩余参数
必要参数,默认参数和可选参数有个共同点:它们表示某一个参数。 有时,想同时操作多个参数,或者并不知道会有多少参数传递进来。 在JavaScript 里,可以使用 arguments
来访问所有传入的参数。
在 TypeScript 里,可以把所有参数收集到一个变量里
function buildName(firstName: string, ...rest: string[]) {
return firstName + ' ' + rest.join(' ')
}
let employeeName = buildName('Joseph', 'Samuel', 'Lucas', 'MacKinzie')
剩余参数会被当做个数不限的可选参数。可以一个都没有,同样也可以有任意个。编译器创建参数数组,名字是在省略号(...)后面给定的名字,可以在函数体内使用这个数组
这个省略号也会在带有剩余参数的函数类型定义上使用到:
function buildName(firstName: string, ...rest: string[]) {
return firstName + ' ' + rest.join(' ')
}
let buildNameFun: (fname: string, ...rest: string[]) => string = buildName
this
学习如何在 JavaScript 里正确使用 this
就好比一场成年礼。由于 TypeScript 是 JavaScript 的超集,TypeScript 程序员也需要弄清 this
工作机制并且当有 bug 的时候能够找出错误所在。 幸运的是,TypeScript 能通知错误地使用了 this
的地方。 如果想了解 JavaScript 里的 this是如何工作的,那么首先阅读 Yehuda Katz 写的 Understanding JavaScript Function Invocation and "this"。 Yehuda 的文章详细的阐述了 this
的内部工作原理,因此这里只做简单介绍
this 和箭头函数
JavaScript里,this
的值在函数被调用的时候才会指定。 这是个既强大又灵活的特点,但是需要花点时间弄清楚函数调用的上下文是什么。 但众所周知,这不是一件很简单的事,尤其是在返回一个函数或将函数当做参数传递的时候
下面看一个例子:
let deck = {
suits: ['hearts', 'spades', 'clubs', 'diamonds'],
cards: Array(52),
createCardPicker: function () {
return function () {
let pickedCard = Math.floor(Math.random() * 52)
let pickedSuit = Math.floor(pickedCard / 13)
return { suit: this.suits[pickedSuit], card: pickedCard % 13 }
}
}
}
let cardPicker = deck.createCardPicker()
let pickedCard = cardPicker()
alert('card: ' + pickedCard.card + ' of ' + pickedCard.suit)
可以看到 createCardPicker
是个函数,并且又返回了一个函数。 如果尝试运行这个程序,会发现它并没有弹出对话框而是报错了。 因为 createCardPicker
返回的函数里的 this
被设置成了 window
而不是 deck
对象。 因为只是独立地调用了 cardPicker()
。 顶级的非方法式调用会将 this
视为 window
。 (注意:在严格模式下,this
为 undefined
而不是 window
)
为了解决这个问题,可以在函数被返回时就绑好正确的 this
。 这样的话,无论之后怎么使用它,都会引用绑定的 deck
对象。 需要改变函数表达式来使用 ECMAScript 6 箭头语法。 箭头函数能保存函数创建时的 this
值,而不是调用时的值:
let deck = {
suits: ['hearts', 'spades', 'clubs', 'diamonds'],
cards: Array(52),
createCardPicker: function () {
// 注意:这里使用箭头函数
return () => {
let pickedCard = Math.floor(Math.random() * 52)
let pickedSuit = Math.floor(pickedCard / 13)
return { suit: this.suits[pickedSuit], card: pickedCard % 13 }
}
}
}
let cardPicker = deck.createCardPicker()
let pickedCard = cardPicker()
alert('card: ' + pickedCard.card + ' of ' + pickedCard.suit)
this 参数
在上述的例子中 this.suits[pickedSuit]
的类型为 any
,这是因为 this
来自对象字面量里的函数表达式。 修改的方法是,提供一个显式的 this
参数。 this
参数是个假的参数,它出现在参数列表的最前面:
function f(this: void) {
// 确保“this”在此独立函数中不可用
}
往例子里添加一些接口,Card
和 Deck
,让类型重用能够变得清晰简单些:
interface Card {
suit: string
card: number
}
interface Deck {
suits: string[]
cards: number[]
createCardPicker(this: Deck): () => Card
}
let deck: Deck = {
suits: ['hearts', 'spades', 'clubs', 'diamonds'],
cards: Array(52),
// NOTE: 函数现在显式指定其被调用方必须是 deck 类型
createCardPicker: function (this: Deck) {
return () => {
let pickedCard = Math.floor(Math.random() * 52)
let pickedSuit = Math.floor(pickedCard / 13)
return { suit: this.suits[pickedSuit], card: pickedCard % 13 }
}
}
}
let cardPicker = deck.createCardPicker()
let pickedCard = cardPicker()
alert('card: ' + pickedCard.card + ' of ' + pickedCard.suit)
现在 TypeScript 知道 createCardPicker
期望在某个 Deck
对象上调用。 也就是说 this
是 Deck
类型的,而非 any
,因此 --noImplicitThis
不会报错了
回调函数里的 this 参数
当将一个函数传递到某个库函数里在稍后被调用时,可能也见到过回调函数里的 this
会报错。 因为当回调函数被调用时,它会被当成一个普通函数调用,this
将为 undefined
。稍做改动,就可以通过 this
参数来避免错误。 首先,库函数的作者要指定this的类型:
interface UIElement {
addClickListener(onclick: (this: void, e: Event) => void): void
}
this: void
意味着 addClickListener
期望 onclick
是一个函数且它不需要一个 this
类型。 然后,为调用代码里的 this
添加类型注解:
class Handler {
info: string
onClickBad(this: Handler, e: Event) {
// 使用此回调将在运行时崩溃
this.info = e.message
}
}
let h = new Handler()
uiElement.addClickListener(h.onClickBad) // Error
指定了 this
类型后,显式声明 onClickBad
必须在 Handler
的实例上调用。 然后 TypeScript 会检测到 addClickListener
要求函数带有 this: void
。 改变 this
类型来修复这个错误:
class Handler {
info: string
onClickGood(this: void, e: Event) {
// 这里不能使用,因为类型是void
console.log('clicked!')
}
}
let h = new Handler()
uiElement.addClickListener(h.onClickGood)
因为 onClickGood
指定了 this
类型为 void
,因此传递 addClickListener
是合法的。 当然了,这也意味着不能使用 this.info
,如果两者都想要,就不得不使用箭头函数了:
class Handler {
info: string
onClickGood = (e: Event) => {
this.info = e.message
}
}
这是可行的,因为箭头函数使用外层的 this
,所以总是可以把它们传给期望 this: void
的函数。缺点是每个 Handler
对象都会创建一个箭头函数。 另一方面,方法只会被创建一次,添加到 Handler
的原型链上。 它们在不同 Handler
对象间是共享的
重载
JavaScript 本身是个动态语言。JavaScript 里函数根据传入不同的参数而返回不同类型的数据是很常见的。
let suits = ['hearts', 'spades', 'clubs', 'diamonds']
function pickCard(x): any {
// 检查是否正在使用对象/数组
// 如果是这样,他们给了我们一副牌,我们来选牌
if (typeof x == 'object') {
let pickedCard = Math.floor(Math.random() * x.length)
return pickedCard
}
// 否则,就让他们挑牌吧
else if (typeof x == 'number') {
let pickedSuit = Math.floor(x / 13)
return { suit: suits[pickedSuit], card: x % 13 }
}
}
let myDeck = [
{ suit: 'diamonds', card: 2 },
{ suit: 'spades', card: 10 },
{ suit: 'hearts', card: 4 }
]
let pickedCard1 = myDeck[pickCard(myDeck)]
alert('card: ' + pickedCard1.card + ' of ' + pickedCard1.suit)
let pickedCard2 = pickCard(15)
alert('card: ' + pickedCard2.card + ' of ' + pickedCard2.suit)
pickCard
方法根据传入参数的不同会返回两种不同的类型。 如果传入的是代表纸牌的对象,函数作用是从中抓一张牌。 如果用户想抓牌,就告诉他抓到了什么牌。 但是这怎么在类型系统里表示呢。
方法是为同一个函数提供多个函数类型定义来进行函数重载。 编译器会根据这个列表去处理函数的调用。 下面来重载pickCard函数
let suits = ['hearts', 'spades', 'clubs', 'diamonds']
function pickCard(x: { suit: string; card: number }[]): number
function pickCard(x: number): { suit: string; card: number }
function pickCard(x): any {
// 检查是否正在使用对象/数组
// 如果是这样,他们给了我们一副牌,我们来选牌
if (typeof x == 'object') {
let pickedCard = Math.floor(Math.random() * x.length)
return pickedCard
}
// 否则,就让他们挑牌吧
else if (typeof x == 'number') {
let pickedSuit = Math.floor(x / 13)
return { suit: suits[pickedSuit], card: x % 13 }
}
}
let myDeck = [
{ suit: 'diamonds', card: 2 },
{ suit: 'spades', card: 10 },
{ suit: 'hearts', card: 4 }
]
let pickedCard1 = myDeck[pickCard(myDeck)]
alert('card: ' + pickedCard1.card + ' of ' + pickedCard1.suit)
let pickedCard2 = pickCard(15)
alert('card: ' + pickedCard2.card + ' of ' + pickedCard2.suit)
这样改变后,重载的 pickCard
函数在调用的时候会进行正确的类型检查
为了让编译器能够选择正确的检查类型,它与 JavaScript 里的处理流程相似。 它查找重载列表,尝试使用第一个重载定义。如果匹配的话就使用这个。因此,在定义重载的时候,一定要把最精确的定义放在最前面
注意,function pickCard(x): any
并不是重载列表的一部分,因此这里只有两个重载:一个是接收对象另一个接收数字。以其它参数调用 pickCard
会产生错误