目录
一.简介
TypeScript中泛型的定义:在定义函数时,不决定函数的参数类型,而是让调用者以参数的形式告知,函数的参数应该是什么样的类型。
使用泛型之前,我们写函数的类型,是这样的:
TypeScript
function identity(arg: any): any {
return arg;
}
尽管使用 any类型可以让我们接受任何类型的 arg 参数,但也让我们丢失了函数返回时的类型信息。如果我们传入一个数字,我们唯一知道的信息是函数可以返回任何类型的值
所以我们需要一种可以捕获参数类型的方式,然后再用它表示返回值的类型
TypeScript
function identity<Type>(arg: Type): Type {
return arg;
}
这个Type 允许我们捕获调用者提供的类型,使得接下来可以使用这个类型,可以看到,返回值也使用了这个类型。这就是泛型。函数的参数和返回值的类型不是固定的,而是由函数的调用者决定的。
我们有两种方式调用上边的泛型函数:
- 传入所有的参数(包括类型参数)
TypeScript
let output = identity<string>("myString"); // output: string
在这里,我们使用 **<>**而不是 **()**包裹了参数,并明确的设置 Type为 string作为函数调用的一个参数;
- 使用类型推导
TypeScript
let output = identity("myString"); // let output: string
这次我们并没有用 <>
明确的传入类型,当编译器看到 myString
这个值,就会自动设置 Type
为它的类型(即 string
)。
二.泛型接口的使用
接口中也可以使用泛型:
TypeScript
// 可以同时传入多个类型
interface User<T, N> {
name: T,
age: N
}
const user: User<string, number> = {
name: 'hsh',
age: 18
}
user.name // ok
user.age // ok
三.泛型约束
有些时候,需要对Type做一些类型约束,以保证顺利通过编译。
假如现在有一个getLength函数:
TypeScript
function getLength<T>(arg: T) {
console.log(arg.length); // Error: 类型"T"上不存在属性"length"
}
上边的写法会直接报错,因为Type可能是任意类型,而如果是number
类型的话,是没有length
属性的,这个时候要对泛型进行类型约束:
TypeScript
// 使用接口约束
interface ILength {
length: number
}
function getLength<T extends ILength>(arg: T) {
console.log(arg.length);
}
getLength('111') // ok
getLength(['111']) // ok
// getLength(111) // Error: 类型"number"的参数不能赋给类型"ILength"的参数
现在这个泛型函数被约束了,它不再适用于所有类型,我们需要传入符合约束条件的值