TypeScript 凭借其强大的类型系统,提供了一项称为泛型的功能,它使开发人员能够编写可重用和类型安全的代码。泛型允许您创建可以处理多种类型(而不是单个类型)的组件。
本文深入探讨了 TypeScript 泛型,提供了详尽的解释和代码示例来说明它们的用法和好处。
什么是泛型?
TypeScript 中的泛型允许编写可处理各种数据类型的代码,同时保持类型安全。它们允许在不牺牲类型检查的情况下创建可重用的组件、函数和数据结构。
泛型由类型参数表示,这些参数充当类型的占位符。这些参数在尖括号 ( <>
) 中指定,可以在整个代码中用于定义变量类型、函数参数、返回类型等。
TypeScript 泛型用例
泛型的基本用法
让我们从泛型函数的简单示例开始:
ts
function identity<T>(arg: T): T {
return arg;
}
const output = identity<string>("hello");
console.log(output); // hello
在此示例中, identity 是一个采用类型参数的泛型函数 T 。参数 arg 的类型为 T ,函数的返回类型也是 T 。调用 identity<string>("hello")
时,类型参数 T 推断为 string ,确保类型安全。
如何使用泛型类
泛型不仅限于函数,它们还可以与类一起使用。请考虑以下泛型 Box 类示例:
ts
class Box<T> {
private value: T;
constructor(value: T) {
this.value = value;
}
getValue(): T {
return this.value;
}
}
const box = new Box<number>(42);
console.log(box.getValue()); // 42
这里, Box 是一个带有类型参数的泛型类 T 。构造函数采用 type 的值,该 getValue 方法返回 type T T 的值。创建 的实例时 Box ,它只能存储和返回 类型的 number 值。
如何对泛型应用约束
有时,您可能希望限制可用于泛型的类型。TypeScript 允许您使用 extends 关键字指定对类型参数的约束。让我们看一个例子
ts
interface Lengthwise {
length: number;
}
function loggingIdentity<T extends Lengthwise>(arg: T): T {
console.log(arg.length);
return arg;
}
const result = loggingIdentity("hello");
console.log(result); // Output: hello
在此示例中,该 loggingIdentity 函数采用一个类型参数, T 该参数必须扩展 Lengthwise 接口,从而确保具有 arg length 属性。此约束允许在不导致编译错误的情况下访问 length 该属性。
如何将泛型与接口一起使用
泛型还可以与接口一起使用,以创建灵活且可重用的定义。请看以下示例:
ts
interface Pair<T, U> {
first: T;
second: U;
}
let pair: Pair<number, string> = { first: 1, second: "two" };
console.log(pair); // Output: { first: 1, second: "two" }
这里, Pair 是一个具有两个类型参数 T 和 的接口,分别表示 first 和 second U 属性的类型。声明 pair 为 Pair<number, string>
时,它强制要求属性必须是数字,并且 first second 属性必须是字符串。
如何将泛型函数与数组一起使用
ts
function reverse<T>(array: T[]): T[] {
return array.reverse();
}
let numbers: number[] = [1, 2, 3, 4, 5];
let reversedNumbers: number[] = reverse(numbers);
console.log(reversedNumbers); // Output: [5, 4, 3, 2, 1]
在此示例中,该 reverse 函数采用 type T 的数组,并返回相同类型的反向数组。通过使用泛型,该函数可以处理任何类型的数组,从而确保类型安全。
如何将泛型约束与 keyof
ts
function getProperty<T, K extends keyof T>(obj: T, key: K): T[K] {
return obj[key];
}
let person = { name: "John", age: 30, city: "New York" };
let age: number = getProperty(person, "age");
console.log(age); // Output: 30
在这里,该 getProperty 函数接受一个 type 的对象和一个 type T K 的键,其中 K 扩展了 T 的键。然后,它从对象返回相应的属性值。此示例演示如何在访问对象属性时使用泛型 keyof 来强制实施类型安全性。
如何使用泛型实用程序函数
ts
function toArray<T>(value: T): T[] {
return [value];
}
let numberArray: number[] = toArray(42);
console.log(numberArray); // Output: [42]
let stringArray: string[] = toArray("hello");
console.log(stringArray); // Output: ["hello"]
该函数将类型的 T 单个值转换为包含该 toArray 值的数组。这个简单的实用函数展示了如何使用泛型来创建可重用的代码,这些代码可以毫不费力地适应不同的数据类型。
如何将泛型接口与函数一起使用
ts
interface Transformer<T, U> {
(input: T): U;
}
function uppercase(input: string): string {
return input.toUpperCase();
}
let transform: Transformer<string, string> = uppercase;
console.log(transform("hello")); // Output: HELLO
在此示例中,我们定义了一个 Transformer 接口,其中包含两个类型参数 T 和 U ,分别表示输入和输出类型。然后,我们声明一个函数 uppercase 并将其分配给 类型的 Transformer<string, string> 变量 transform 。这演示了如何使用泛型来定义函数的灵活接口。
总结
无论是函数、类还是接口,泛型都为构建可伸缩和可维护的 TypeScript 应用程序提供了可靠的机制。理解和掌握泛型可以显著提高您编写高效且无错误的代码的能力。