TypeScript 泛型如何工作

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 应用程序提供了可靠的机制。理解和掌握泛型可以显著提高您编写高效且无错误的代码的能力。

参考资料

-How TypeScript Generics Work -- Explained with Examples

相关推荐
码客前端3 分钟前
理解 Flex 布局中的 flex:1 与 min-width: 0 问题
前端·css·css3
Komorebi゛4 分钟前
【CSS】圆锥渐变流光效果边框样式实现
前端·css
工藤学编程16 分钟前
零基础学AI大模型之CoT思维链和ReAct推理行动
前端·人工智能·react.js
徐同保17 分钟前
上传文件,在前端用 pdf.js 提取 上传的pdf文件中的图片
前端·javascript·pdf
怕浪猫18 分钟前
React从入门到出门第四章 组件通讯与全局状态管理
前端·javascript·react.js
博主花神18 分钟前
【React】扩展知识点
javascript·react.js·ecmascript
欧阳天风25 分钟前
用setTimeout代替setInterval
开发语言·前端·javascript
EndingCoder29 分钟前
箭头函数和 this 绑定
linux·前端·javascript·typescript
郑州光合科技余经理29 分钟前
架构解析:同城本地生活服务o2o平台海外版
大数据·开发语言·前端·人工智能·架构·php·生活
沐墨染31 分钟前
大型数据分析组件前端实践:多维度检索与实时交互设计
前端·elementui·数据挖掘·数据分析·vue·交互