基本使用
泛型使用尖括号 来表示,并在定义函数、类或接口时指定类型参数。
const city = <T>(item: T): T => {
return item; // T类似一个变量保证入参和出参类型一致。
};
let result = city<number>(42);
let result2 = city<string>("Southern Wind");
console.log(result); // 42
console.log(result2); // Southern Wind
这里可能就有疑问了,为什么要用泛型?原因:泛型可以让我们在编写代码时更加灵活和可重用,同时也可以避免类型错误的发生。
const city = (item: number | string): number | string => {
return item;
};
比如接受 number 和 string 多个类型的参数,我们可以使用泛型来解决这个问题
泛型类型
const reverse = <T>(array: T[]): T[] => {
return array.reverse();
};
const arr: number[] = [1, 2, 3, 4, 5, 6];
const reversedArr = reverse(arr);
console.log(reversedArr); // (6) [6, 5, 4, 3, 2, 1]
let string: string[] = ["a", "b", "c", "d", "e"];
let reversedString = reverse(string);
console.log(reversedString); // (5) ["e", "d", "c", "b", "a"]
泛型类型允许我们创建可以适用于多种类型的类,类中的成员可以使用泛型类型进行声明和使用
多类型集合,泛型如何处理?
- 返回类型集合
const citys = <T, U>(item: T, msg: U): [T, U] => {
return [item, msg];
};
console.log(citys<number, string>(123, "456"));
- 泛型接口
可以写一个接口,把接口内的参数变为泛型
例如:
interface IFromData<T, U, V> {
data: T;
msg: U;
status: V;
code: V;
}
const per = <T, U, V>(
data: T,
msg: U,
status: V,
code: V
): IFromData<T, U, V> => {
let result: IFromData<T, U, V> = {
data,
msg,
status,
code,
};
return result;
};
let s = per<object, string, number>(
{ name: "Southern Wind", age: 18 },
"欢迎",
200,
0
);
console.log(s);
常用操作符
:extends
用来约束泛型类型,比如我想写一个函数,用来检查对象中是否存在某个 key,如果存在则返回 true,否则返回 false,我们可以使用泛型约束来实现这个功能(此处后面会提到)
泛型约束可以用来限制泛型类型的范围,使其满足特定条件。
:keyof
用来获取对象的 key,比如我想写一个函数,用来获取对象的 key,我们可以使用泛型约束来实现这个功能
interface Person {
name: string;
age: number;
isMarray: boolean;
}
type per1 = keyof Person; // string | number | boolean
class Stack<T> {
private items: T;
constructor(items: T) {
this.items = items;
}
getItems(): T {
return this.items;
}
}
let stack = new Stack<number>(123332);
console.log(stack.getItems()); // 123332
let stack2 = new Stack<string>("Southern Wind");
console.log(stack2.getItems()); // Southern Wind
使用泛型参数,可以在类的定义中引入类型变量来表示未知的类型。这样我们就可以在类实例化时指定具体的类型,从而创建适用于不同类型数据的类的实例。
泛型约束
假设length属性是可以用的,那么我们使用函数输出参数长度就会出现问题
const indentify = <T>(args: T):T => {
console.log(args.length); // // 类型“T”上不存在
return args;
}
indentify([1,2,3,4,5,6,7,8])
出现这个问题是因为ts
不能识别T
上是否存在某个属性,我们需要对其定义一个类型,类型变量为extends
interface ILength{
length: number;
}
const indentify = <T extends ILength>(args: T):T => {
console.log(args.length); // 8
return args;
}
indentify([1,2,3,4,5,6,7,8])
当然也可以这么操作
const indentify = <T>(args: T[]):T[] => {
console.log(args.length); // 8
return args;
}
let arr = [1,2,3,4,5,6,7,8]
console.log(indentify(arr)); // [1, 2, 3, 4, 5, 6, 7, 8]
const indentify2 = <T>(args: Array<T>):Array<T> => {
console.log(args.length); // 8
return args;
}
let arr2 = [9,10,11,12,13,14,15,16]
console.log(indentify2(arr2)); // [9,10,11,12,13,14,15,16]
也可以通过keyof
来确认对象上的键是否存在
interface IK {
name: string;
sex: string;
age: number;
address: string;
}
// 通过 keyof 操作符,我们就可以获取指定类型的所有键,之后我们可以对结合extends约束
const indentify = <IPerson, K extends keyof IK>(obj: IK, key: K): IK[K] => {
return obj[key];
};
// 使用 indentify 函数
const person: IK = {
name: "Alice",
sex: "female",
age: 30,
address: "123 Main St"
};
const nameKey: keyof IK = "name";
const nameValue = indentify(person, nameKey);
console.log("Name:", nameValue); // Name: Alice
确保属性是否存在
比如我想写一个获取对象长度的函数 getLength,但不是所有的对象都有 length 属性,所以我们需要使用泛型约束来确保传入的对象具有 length 属性
/* function getLength<T extends { length: number }>(obj: T): number {
return obj.length; 或
} */
interface Length {
length: number;
}
function getLength<T extends Length>(obj: T): number {
return obj.length;
}
let obj = "Hello";
let obj2 = [1, 2, 3, 4, 5, 6];
console.log(getLength(obj)); // 5
console.log(getLength(obj2)); // 6
检查对象的 key
比如我想写一个函数,用来检查对象中是否存在某个 key,如果存在则返回 true,否则返回 false,我们可以使用泛型约束来实现这个功能
function hasKey<T extends object, K extends keyof T>(obj: T, key: K): boolean {
return key in obj;
}
let obj3 = {
name: "Southern Wind",
age: 20,
gender: "male",
};
console.log(hasKey(obj3, "name")); // true
泛型接口
使用泛型接口可以创建适用于多种类型的接口,接口中的成员可以使用泛型类型进行声明和使用
interface Transformers<T, U> {
transform(input: T): U;
}
class StringToNumberTransformer implements Transformers<string, number> {
transform(input: string): number {
return parseFloat(input);
}
}
let transformer = new StringToNumberTransformer();
let results = transformer.transform("3.14");
console.log(typeof results, results); // number 3.14