TypeScript技术系列:深入理解函数类型与其应用
前言
欢迎来到TypeScript
技术系列的第4讲!在前几篇中,我们已经探讨了函数定义的基本类型推断特性。今天,我们将深入了解TypeScript
中的函数类型,包括函数的参数和返回值类型、函数重载、this的类型等等。本文将通过详细的解释和示例代码,帮助你全面掌握这些概念。
1. 函数类型的基础
在JavaScript
中,函数是非常核心的构建块,用于封装逻辑和处理各种操作。在TypeScript
中,函数不仅保持了 JavaScript
的所有特性,还增加了强类型系统的支持,使得函数的定义和调用更加严谨。
1.1 函数定义
TypeScript
支持两种主要的函数定义方式:函数声明和箭头函数。来看一下这两种方式的基本语法:
// 函数声明
function add(a: number, b: number): number {
return a + b;
}
// 箭头函数
const multiply = (x: number, y: number): number => x * y;
在这段代码中,add
是一个传统的函数声明,接受两个数字参数并返回它们的和。multiply
是一个箭头函数,实现了同样的功能。TypeScript
的类型注解:number
用于指定函数的参数和返回值类型。
1.2 返回值类型
在JavaScript
中,函数可能会返回undefined
,但在TypeScript
中,需要明确声明返回值类型。如果函数没有返回值,可使用void
类型来表示。
function logMessage() {
console.log(message);
}
console.log(logMessage()) // undefined
需要注意的是,在TypeScript
中,如果显式声明函数的返回值类型为undfined
,将会得到如下所示的错误提醒。
function logMessage(): undefined {
// TS2355: A function whose declared type is neither 'void' nor 'any' must return a value
// TODO
}
此时,正确的做法是使用前面文章介绍的void
类型来表示函数没有返回值的类型(这是“废柴”void
类型唯一有用的场景),示例如下:
function logMessage(): void {}
但是很多时候,不必或者不能显式地指明返回值的类型,这就涉及可缺省和可推断的返回值类型的讲解。
1.3 可缺省和可推断的返回值类型
幸运的是,函数返回值的类型可以在TypeScript
中被推断出来,即可缺省。
函数内是一个相对独立的上下文环境,可以根据入参对值计算,并返回新的值。从类型层面看,也可以通过类型推断计算入参的类型,并返回新的类型,示例如下:
function fn(a: string, b: number) {
const nums = [b];
const strs = [a]
return {
nums,
strs
} // 返回 { nums: number[]; strs: string[] } 的类型
}
一般情况下,TypeScript
中的函数返回值类型是可以缺省和推断出来的,但是有些特例需要显式声明返回值类型,比如Generator函数的返回值。
1.4 Generator函数的返回值
ES6
中新增的Generator函数在TypeScript
中也有对应的类型定义。
Generator函数返回的是一个Iterator
迭代器对象,可以使用Generator
的同名接口泛型或者Iterator
的同名接口泛型表示返回值的类型(Generator
类型继承了Iterator
类型),示例如下:
type Type = boolean;
type ReturnType = string;
type NextType = number;
function *gen(): Generator<Type, ReturnType, NextType> {
const nextValue = yield true; // nextValue 类型是 number,yield 后必须是 boolean 类型
return `${nextValue}`; // 必须返回 string 类型
}
2. 参数类型
了解了定义函数的基本语法以及返回值类型后,再来详细看一下可选参数、默认参数、剩余参数的几个特性。
2.1 可选参数和默认参数
在实际工作中,可能经常碰到函数参数可传可不传的情况,当然TypeScript
也支持这种函数类型表达,如下代码所示:
function fn(str?: string) {
return str;
}
fn(); // undefined
fn('hello world'); // hello world
在上述代码中,在类型标注的:
前添加?
表示fn
函数的参数str
就是可缺省的。
也就是说参数str
的类型可能是undefined
类型或者是string
类型,那是不是意味着可缺省和类型是undefined
等价呢?来看看以下的示例:
function fn1(str?: string) {
console.log(str);
}
function fn2(str: string | undefined) {
console.log(str);
}
fn1();
fn1(undefined);
fn2(); // ts(2554) Expected 1 arguments, but got 0
fn2(undefined);
答案显而易见,这里的?:
表示参数可以缺省、可以不传,也就是说调用函数时,可以不显式传入参数。但是,如果声明了参数类型为xxx | undefined
,就表示函数参数是不可缺省且类型必须是xxx
或者undfined
。
因此,在上述代码中,fn1
函数如果不显示传入函数的参数,TypeScript
就会报一个ts(2554)
的错误,即函数需要1个参数,但是只传入了0个参数。
在ES6
中支持函数默认参数的功能,而TypeScript
会根据函数的默认参数的类型来推断函数参数的类型,示例如下:
function fn(str = 'hello') {
console.log(x);
}
fn(); // => 'hello'
fn('hi'); // => 'hi'
fn(1); // ts(2345) Argument of type '1' is not assignable to parameter of type 'string | undefined'
在上述示例中,根据函数的默认参数'hello'
,TypeScript
推断出了str
的类型为string | undefined
。
当然,对于默认参数,TypeScript
也可以显式声明参数的类型。不过,此时的默认参数只起到参数默认值的作用,如下代码所示:
function fn1(x: string = 'hello') {
console.log(x);
}
// ts(2322) Type 'string' is not assignable to type 'number'
function fn2(x: number = 'hello') {
console.log(x);
}
fn2();
fn2(1);
fn2('1'); // ts(2345) Argument of type '"1"' is not assignable to parameter of type 'number | undefined'
上例函数fn2
中,显式声明了函数参数str
的类型为number
,表示函数参数str
的类型可以不传或者是number
类型。因此,如果将默认值设置为字符串类型,编译器就会抛出一个ts(2322)
的错误。
2.2 剩余参数
在ES6
中,JavaScript
支持函数参数的剩余参数,它可以把多个参数收集到一个变量中。同样,在TypeScript
中也支持这样的参数类型定义,如下代码所示:
function sum(...nums: number[]) {
return nums.reduce((a, b) => a + b, 0);
}
sum(1, 2); // => 3
sum(1, 2, 3); // => 6
sum(1, '2'); // ts(2345) Argument of type 'string' is not assignable to parameter of type 'number'
在上述代码中,sum
是一个求和的函数,...nums
将函数的所有参数收集到了变量nums
中,而nums
的类型应该是number[]
,表示所有被求和的参数是数字类型。因此,sum(1, '2')
抛出了一个ts(2345)
的错误,因为参数'2'
并不是number
类型。
如果将函数参数nums
聚合的类型定义为(number | string)[]
,如下代码所示:
function sum(...nums: (number | string)[]): number {
return nums.reduce<number>((a, b) => a + Number(b), 0);
}
sum(1, '2', 3); // 6
那么,函数的每一个参数的类型就是联合类型number | string
,因此sum(1, '2', 3)
的类型检查也就通过了。
好了,介绍完函数的参数,再来了解一下函数中另外一个重要的知识点this。
2.3 this
众所周知,在JavaScript
中,函数this
的指向一直是一个令人头痛的问题。因为this
的值需要等到函数被调用时才能被确定,更别说通过一些方法还可以改变this
的指向。也就是说this
的类型不固定,它取决于执行时的上下文。
但是,使用了TypeScript
后,就不用担心这个问题了。通过指定this
的类型,当错误使用了this
,TypeScript
就会提示,如下代码所示:
function say() {
console.log(this.name); // ts(2683) 'this' implicitly has type 'any' because it does not have a type annotation
}
say();
在上述代码中,如果直接调用say
函数,this
应该指向全局window
或global(Node 中)
。但是,在strict
模式下的TypeScript
中,它会提示this
的类型是any
,此时就需要手动显式指定类型了。
那么,在TypeScript
中,应该如何声明this
的类型呢?
在TypeScript
中,只需要在函数的第一个参数中��明this
指代的对象(即函数被调用的方式)即可,比如最简单的作为对象的方法的this
指向,如下代码所示:
function say(this: Window, name: string) {
console.log(this.name);
}
window.say = say;
window.say('hi');
const obj = { say };
obj.say('hi');
在上述代码中,在window
对象上增加say
的属性为函数say
。那么调用window.say()
时,this
指向即为window
对象。
调用obj.say()
后,此时TypeScript
检测到this
的指向不是window
,于是抛出了如下所示的一个ts(2684)
错误。
say('maybe'); // ts(2684) The 'this' context of type 'void' is not assignable to method's
// 'this' of type 'Window'
需要注意的是,如果直接调用say()
,this
实际上应该指向全局变量window
,但是因为TypeScript
无法确定say
函数被谁调用,所以将this
的指向默认为void
,也就提示了一个ts(2684)
错误。
此时,可以通过调用window.say()
来避免这个错误,这也是一个安全的设计。因为在JavaScript
的严格模式下,全局作用域函数中this
的指向是undefined
。
同样,定义对象的函数属性时,只要实际调用中this
的指向与指定的this
指向不同,TypeScript
就能发现this
指向的错误,示例代码如下:
interface Person {
name: string;
say(this: Person): void;
}
const person: Person = {
name: 'maybe',
say() {
console.log(this.name);
},
};
const fn = person.say;
fn(); // ts(2684) The 'this' context of type 'void' is not assignable to method's 'this' of type 'Person'
介绍完函数中this
的指向和类型后,再来了解一下它的另外一个特性函数多态(函数重载)。
3. 函数重载
函数重载允许为同一个函数定义多个签名,以处理不同的参数和返回值类型。
// 函数重载示例
function parseValue(value: string): number;
function parseValue(value: number): string;
function parseValue(value: string | number): number | string {
if (typeof value === "string") {
return parseFloat(value);
} else {
return value.toString();
}
}
console.log(parseValue("42")); // 42
console.log(parseValue(42)); // "42"
在这个例子中,为parseValue
函数定义了两个重载签名,一个处理字符串参数,另一个处理数字参数。函数的实现部分处理了这些不同的参数类型,并返回相应的值。
4. 类型谓词(Type Predicates)
类型谓词用于在TypeScript
中执行类型缩小,通常用于自定义类型守卫。
// 类型谓词示例
function isString(value: any): value is string {
return typeof value === "string";
}
function handleValue(value: any) {
if (isString(value)) {
console.log(`String value: ${value}`);
} else {
console.log(`Not a string.`);
}
}
handleValue("hello"); // String value: hello
handleValue(42); // Not a string.
在这个示例中,isString
函数是一个类型谓词,它检查一个值是否为字符串,并在handleValue
函数中使用来缩小value
的类型。
总结
本文通过详尽的解释和示例代码,帮助你全面理解TypeScript
中函数类型的各种特性和应用。掌握这些概念可以显著提高你在TypeScript
开发中的效率和代码质量。在下一篇文章中,我们将继续探索TypeScript
的类,深入了解面向对象编程的更多细节。
后语
小伙伴们,如果觉得本文对你有些许帮助,点个👍或者➕个关注再走吧^_^ 。另外如果本文章有问题或有不理解的部分,欢迎大家在评论区评论指出,我们一起讨论共勉。
转载自:https://juejin.cn/post/7397292988642115636