目录
准备工作" >准备工作
TypeScript里的泛型是个啥" >TypeScript里的泛型是个啥
在VSCode中配置TypeScript" >在VSCode中配置TypeScript
找到问题" >找到问题
理解中心思想" >理解中心思想
使用泛型" >使用泛型
泛型约束" >泛型约束
为什么是泛型" >为什么是泛型
其他资源" >其他资源
结论" >结论
首页 web前端 js教程 了解TypeScript中泛型(Generics)的概念和用法

了解TypeScript中泛型(Generics)的概念和用法

Dec 31, 2020 pm 05:18 PM
javascript typescript 泛型

了解TypeScript中泛型(Generics)的概念和用法

本文介绍TypeScript中泛型(Generics)的概念和用法,它为什么重要,及其使用场景。我们会以一些清晰的例子,介绍其语法,类型和如何构建参数。你可以在你的集成开发环境中跟着实践。

准备工作

要从本文中跟着学习的话,你需要在电脑上准备以下东西:

  • 安装Node.js:你可以运行命令行检查Node是否安装好了。
node -v
登录后复制
  • 安装Node Package Manager: 通常安装Node时,它会顺带安装好所需版本的NPM。
  • 安装TypeScript:如果你安装好了Node Package Manager,你可以用以下命令在本机的全局环境安装TypeScript。
  • npm install -g typescript
    登录后复制
    集成开发环境:本文将使用微软团队开发的Visual Studio Code。可以在这里下载。进入其下载的目录,并按照提示进行安装。记得选择“添加打开代码”(Add open with code)选项,这样你就可以在本机从任何位置轻松打开VS Code了。

本文是写给各层次的TypeScript开发人员的,包括但并不只是初学者。 这里给出了设置工作环境的步骤,是为了照顾那些TypeScript和Visual Studio Code的新手们。

TypeScript里的泛型是个啥

在TypeScript中,泛型是一种创建可复用代码组件的工具。这种组件不只能被一种类型使用,而是能被多种类型复用。类似于参数的作用,泛型是一种用以增强类(classes)、类型(types)和接口(interfaces)能力的非常可靠的手段。这样,我们开发者,就可以轻松地将那些可复用的代码组件,适用于各种输入。然而,不要把TypeScript中的泛型错当成any类型来使用——你会在后面看到这两者的不同。

类似C#和Java这种语言,在它们的工具箱里,泛型是创建可复用代码组件的主要手段之一。即,用于创建一个适用于多种类型的代码组件。这允许用户以他们自己的类使用该泛型组件。

在VSCode中配置TypeScript

在计算机中创建一个新文件夹,然后使用VS Code 打开它(如果你跟着从头开始操作,那你已经安装好了)。

在VS Code中,创建一个app.ts文件。我的TypeScript代码都会放在这里面。

把下面打日志的代码拷贝到编辑器中:

console.log("hello TypeScript");
登录后复制

按下F5键,你会看到一个像这样的launch.json文件:

{
  // Use IntelliSense to learn about possible attributes.
  // Hover to view descriptions of existing attributes.
  // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
  "version": "0.2.0",
  "configurations": [
    {
      "type": "node",
      "request": "launch",
      "name": "TypeScript",
      "program": "${workspaceFolder}\\app.ts",
      "outFiles": [
        "${workspaceFolder}/**/*.js"
      ]
    }
  ]
}
登录后复制

里面的name字段的值,本来是Launch Program,我把它改成了TypeScript。你可以把它改成其他值。

点击Terminal Tab,选择Run Tasks,再选择一个Task Runner:"TypeScript Watch Mode",然后会弹出一个tasks.json文件,把它改成下面像这样:

 {
  // See https://go.microsoft.com/fwlink/?LinkId=733558
  // for the documentation about the tasks.json format
  "version": "2.0.0",
  "tasks": [
  {
   "label": "echo",
   "type": "shell",
   "command": "tsc",
   "args": ["-w", "-p","."],
   "problemMatcher": [
    "$tsc-watch"
    ],
   "isBackground": true
   }
  ]
 }
登录后复制

app.ts所在的目录,创建另一个文件tsconfig.json。把下面的代码拷贝进去:

{
  "compilerOptions": {
    "sourceMap": true
  }
}
登录后复制

这样,Task Runner就可以把TypeScript编译成JavaScript,并且可监听到文件的变化,实时编译。

再次点击Ternimal标签,选择Run Build Task,再选择tsc: watch - tsconfig.json,可以看到终端出现的信息:

[21:41:31] Starting compilation in watch mode…
登录后复制

你可以使用VS Code的调试功能编译TypeScript文件。  

1.png

设置好了开发环境,你就可以着手处理TypeScript泛型概念相关的问题了。

找到问题

TypeScript中不建议使用any类型,原因有几点,你可以在本文看到。其中一个原因,就是调试时缺乏完整的信息。而选择VS Code作为开发工具的一个很好的理由,就是它带来的基于这些信息的智能感知。

如果你有一个类,存储着一个集合。有方法向该集合里添加东西,也有方法通过索引获取集合里的东西。像这样:

class Collection {
  private _things: string[];
  constructor() {
    this._things = [];
  }
  add(something: string) {
    this._things.push(something);
  }
  get(index: number): string {
    return this._things[index];
  }
}
登录后复制

你可以很快辨识出,此集合被显示定义为一个string类型的集合,显然是不能在其中使用number的。如果想要处理number的话,可以创建一个接受number而不是string的集合。着是一个不错的选择,但有一个很大的缺点——代码重复。代码重复,最终会导致编写和调试代码的时间增多,并且降低内存的使用效率。

另一个选择,是使用any类型代替string类型定义刚才的类,像下面这样:

class Collection {
  private _things: any[];
  constructor() {
    this._things = [];
  }
  add(something: any) {
    this._things.push(something);
  }
  get(index: number): any {
    return this._things[index];
  }
}
登录后复制

此时,该集合支持你给出的任何类型。如果你创建像这样的逻辑构建此集合的话:

let Stringss = new Collection();
Stringss.add("hello");
Stringss.add("world");
登录后复制

这添加了字符串"hello"和"world"到集合中,你可以打出像length这样的属性,返回任意一个集合元素的长度。  

console.log(Stringss.get(0).length);
登录后复制

字符串"hello"有五个字符,运行TypeScript代码,你可以在调试模式下看到它。  

2.png

请注意,当你鼠标悬停在length属性上时,VS Code的智能感知没有提供任何信息,因为它不知道你选择使用的确切类型。当你像下面这样,把其中一个添加的元素修改为其他类型时,比如number,这种不能被智能感知到的情况会体现得更加明显:

let Strings = new Collection();
Strings.add(001);
Strings.add("world");
console.log(Strings.get(0).length);
登录后复制

你打出一个undefined的结果,仍然没有什么有用信息。如果你更进一步,决定打印string的子字符串——它会报运行时错误,但不指不出任何具体的内容,更重要的是,编译器没有给出任何类型不匹配的编译时错误。  

console.log(Stringss.get(0).substr(0,1));
登录后复制

3.png

这仅仅是使用any类型定义该集合的一种后果罢了。

理解中心思想

刚才使用any类型导致的问题,可以用TypeScript中的泛型来解决。其中心思想是类型安全。使用泛型,你可以用一种编译器能理解的,并且合乎我们判断的方式,指定类、类型和接口的实例。正如在其他强类型语言中的情况一样,用这种方法,就可以在编译时发现你的类型错误,从而保证了类型安全。

泛型的语法像这样:

function identity<T>(arg: T): T {
  return arg;
}
登录后复制

你可以在之前创建的集合中使用泛型,用尖括号括起来。  

class Collection<T> {
  private _things: T[];
  constructor() {
    this._things = [];
  }
  add(something: T): void {
    this._things.push(something);
  }
  get(index: number): T {
    return this._things[index];
  }
}
let Stringss = new Collection<String>();
Stringss.add(001);
Stringss.add("world");
console.log(Stringss.get(0).substr(0, 1));
登录后复制

如果将带有尖括号的新逻辑复制到代码编辑器中,你会立即注意到"001"下的波浪线。这是因为,TypeScript现在可以从指定的泛型类型推断出001不是字符串。在T出现的地方,就可以使用string类型,这就实现了类型安全。本质上,这个集合的输出可以是任何类型,但你指明了它应该是string类型,所以编译器推断它就是string类型。这里使用的泛型声明是在类级别,它也可以在其他级别定义,如静态方法级别和实例方法级别,你稍后会看到。

使用泛型

你可以在泛型声明中,包含多个类型参数,它们只需要用逗号分隔,像这样:

class Collection<T, K> {
  private _things: K[];
  constructor() {
    this._things = [];
  }
  add(something: K): void {
    this._things.push(something);
  }
  get(index: number): T {
    console.log(index);
  }
}
登录后复制

声明时,类型参数也可以在函数中显式使用,比如:

class Collection {
  private _things: any[];
  constructor() {
    this._things = [];
  }
  add<A>(something: A): void {
    this._things.push(something);
  }
  get<B>(index: number): B {
    return this._things[index];
  }
}
登录后复制

因此,当你要创建一个新的集合时,在方法级别声明的泛型,现在也会在方法调用级别中被指示,像这样:  

let Stringss = new Collection();
Stringss.add<string>("hello");
Stringss.add("world");
登录后复制

你还可注意到,在鼠标悬停时,VS Code智能感知能够推断出第二个add函数调用仍然是string类型。

泛型声明同样适用于静态方法:

static add<A>(something: A): void {
  _things.push(something);
}
登录后复制

虽然初始化静态方法时,可使用泛型类型,但是,对初始化静态属性则不能。

泛型约束

现在,你已经对泛型有比较好的认识,是时候提到泛型的核心缺点及其实用的解决方案了。使用泛型,许多属性的类型都能被TypeScript推断出来,然而,在某些TypeScript不能做出准确推断的地方,它不会做任何假设。为了类型安全,你需要将这些要求或者约束定义为接口,并在泛型初始化中继承它们。

如果你有这样一个非常简单的函数:

function printName<T>(arg: T) {
  console.log(arg.length);
  return arg;
}
printName(3);
登录后复制

因为TypeScript无法推断出arg参数是什么类型,不能证明所有类型都具有length属性,因此不能假设它是一个字符串(具有length属性)。所以,你会在length属性下看到一条波浪线。如前所述,你需要创建一个接口,让泛型的初始化可以继承它,以便编译器不再报警。  

interface NameArgs {
  length: number;
}
登录后复制

你可以在泛型声明中继承它:

function printName<T extends NameArgs>(arg: T) {
  console.log(arg.length);
  return arg;
}
登录后复制

这告诉TypeScript,可使用任何具有length属性的类型。 定义它之后,函数调用语句也必须更改,因为它不再适用于所有类型。 所以它应看起来是这样:

printName({length: 1, value: 3});
登录后复制

这是一个很基础的例子。但理解了它,你就能看到在使用泛型时,设置泛型约束是多么有用。

为什么是泛型

一个活跃于Stack Overflow社区的成员,Behrooz,在后续内容中很好的回答了这个问题。在TypeScript中使用泛型的主要原因是使类型,类或接口充当参数。 它帮助我们为不同类型的输入重用相同的代码,因为类型本身可用作参数。

泛型的一些好处有:

  • 定义输入和输出参数类型之间的关系。比如
function test<T>(input: T[]): T {
  //… 
}
登录后复制

允许你确保输入和输出使用相同的类型,尽管输入是用的数组。

  • 可使用编译时更强大的类型检查。在上诉示例中,编译器让你知道数组方法可用于输入,任何其他方法则不行。
  • 你可以去掉不需要的强制类型转换。比如,如果你有一个常量列表:
Array<Item> a = [];
登录后复制

变量数组时,你可以由智能感知访问到Item类型的所有成员。

其他资源

结论

你已经看完了泛型概念的概述,并看到了各种示例来帮助揭示它背后的思想。 起初,泛型的概念可能令人困惑,我建议,把本文再读一遍,并查阅本文所提供的额外资源,帮助自己更好地理解。泛型是一个很棒的概念,可以帮助我们在JavaScript中,更好地控制输入和输出。请快乐地编码吧!

更多编程相关知识,请访问:编程教学!!

以上是了解TypeScript中泛型(Generics)的概念和用法的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳图形设置
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您听不到任何人,如何修复音频
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解锁Myrise中的所有内容
4 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

泛型函数在Golang中解决可变参数类型的问题吗? 泛型函数在Golang中解决可变参数类型的问题吗? Apr 16, 2024 pm 06:12 PM

Go中的泛型函数解决了可变参数类型的问题:泛型函数允许使用类型参数,在运行时指定。这使得编写可以处理不同类型参数的函数成为可能。例如,Max函数是一个泛型函数,它接受两个可比较参数并返回较大值。通过使用泛型函数,我们可以编写更灵活通用的代码,可处理不同类型的参数。

golang中泛型的具体应用场景 golang中泛型的具体应用场景 May 04, 2024 am 11:45 AM

泛型在Go中的应用场景:集合操作:创建适用于任何类型的集合操作,例如过滤。数据结构:编写通用的数据结构,如队列,栈和映射,可存储和操作各种类型的数据。算法:编写通用的算法,如排序,搜索和归约,可处理不同类型的数据。

Java 函数泛型的上限和下限是什么?如何使用? Java 函数泛型的上限和下限是什么?如何使用? Apr 26, 2024 am 11:45 AM

Java函数泛型允许设置上限和下限。上限(extends)指定函数接受或返回的数据类型必须是指定类型的子类型,例如。下限(super)指定函数接受或返回的数据类型必须是指定类型的超类型,例如。泛型使用可提高代码的可重用性和安全性。

泛型函数在Golang中的限制是什么? 泛型函数在Golang中的限制是什么? Apr 16, 2024 pm 05:12 PM

Go泛型函数的限制:仅支持类型参数,不支持值参数。不支持函数递归。不能显式指定类型参数,由编译器推断。

Golang泛型对函数签名和参数的影响是什么? Golang泛型对函数签名和参数的影响是什么? Apr 17, 2024 am 08:39 AM

泛型对Go函数签名和参数的影响包括:类型参数:函数签名可包含类型参数,指定函数可使用的类型。类型约束:类型参数可具有约束,指定其必须满足的条件。参数类型推断:编译器可推断未指定类型参数的类型。指定类型:可显式指定参数类型以调用泛型函数。这提高了代码的可重用性和灵活性,允许编写可与多种类型一起使用的函数和类型。

Java 枚举类型如何与泛型配合使用? Java 枚举类型如何与泛型配合使用? May 04, 2024 am 08:36 AM

Java中枚举类型与泛型的结合:声明带泛型的枚举时需添加尖括号,T为类型参数。创建泛型类时,同样需添加尖括号,T为可存储任何类型的类型参数。此结合提高代码灵活性、类型安全性,并简化代码。

golang可变参数是否可以用于泛型函数? golang可变参数是否可以用于泛型函数? Apr 29, 2024 pm 02:06 PM

在Go中,变参可以用于泛型函数,允许创建可接受可变数量参数且适用于多种类型的泛型函数。例如,您可以创建泛型函数Mode,它查找给定列表中出现次数最多的元素:Mode接受可变数量的T类型元素。它通过为每个元素创建counts来统计元素计数。然后它找出出现次数最多的元素,并将其作为mode返回。在main函数中,可以为字符串列表和整数列表调用Mode函数,它将分别返回出现次数最多的字符串和数字。

如何在 Java 泛型方法中限制类型参数? 如何在 Java 泛型方法中限制类型参数? Apr 30, 2024 pm 01:30 PM

为了在Java泛型方法中限制类型参数,需使用语法,其中Bound为类型或接口。如此,参数仅接受继承自Bound类型或实现Bound接口的类型。例如,限制T为可与自身比较的类型。

See all articles