如何通俗的解释TypeScript 泛型

作者:浅笑· 时间:2024-04-10 16:18:31 

概述

在 TypeScript 中我们会使用泛型来对函数的相关类型进行约束。这里的函数,同时包含 class 的构造函数,因此,一个类的声明部分,也可以使用泛型。那么,究竟什么是泛型?如果通俗的理解泛型呢?

什么是泛型

泛型(Generics)是指在定义函数、接口或类的时候,不预先指定具体的类型,而在使用的时候再指定类型的一种特性。

通俗的解释,泛型是类型系统中的“参数”,主要作用是为了类型的重用。从上面定义可以看出,它只会用在函数、接口和类中。它和js程序中的函数参数是两个层面的事物(虽然意义是相同的),因为 typescript 是静态类型系统,是在js进行编译时进行类型检查的系统,因此,泛型这种参数,实际上是在编译过程中的运行时使用。之所以称它为“参数”,是因为它具备和函数参数一模一样的特性。


function increse(param) {
 // ...
}

而类型系统中,我们如此使用泛型:


function increase<T>(param: T): T {
 //...
}

当 param 为一个类型时,T 被赋值为这个类型,在返回值中,T 即为该类型从而进行类型检查。

编译系统

要知道 typescript 本身的类型系统也需要编程,只不过它的编程方式很奇怪,你需要在它的程序代码中穿插 js代码(在 ts 代码中穿插 js 代码这个说法很怪,因为我们直观的感觉是在 js 代码中夹杂了 ts 代码)。

编程中,最重要的一种形式就是函数。在 typescript 的类型编程中,你看到函数了吗?没有。这是因为,有泛型的地方就有函数,只是函数的形式被 js 代码给割裂了。typescript 需要进行编译后得到最终产物。编译过程中要做两件事,一是在内存中运行类型编程的代码,从而形成类型检查体系,也就是说,我们能够对 js 代码进行类型检查,首先是 typescript 编译器运行 ts 编程代码后得到了一个运行时的检查系统本文来自否子戈的播客,运行这个系统,从而对穿插在其中的 js 代码进行类型断言;二是输出 js,输出过程中,编译系统已经运行完了类型编程的代码,就像php代码中 echo js 代码一样,php代码已经运行了,显示出来的是 js 代码。

从这个角度看 typescript,你或许更能理解为什么说它是JavaScript的超集,为什么它的编译结果是 js。

通俗的理解泛型

既然我们理解了 ts 编译系统的逻辑,那么我们就可以把类型的编程和 js 本身的业务编程在情感上区分开。我们所讲的“泛型”,只存在于类型编程的部分,这部分代码是 ts 的编译运行时代码。

我们来看下一个简单的例子:


function increase<T>(param: T): T {
 //...
}

这段代码,如果我们把 js 代码区分开,然后用类型描述文本来表示会是怎样?


// 声明函数 @type,参数为 T,返回结果为 (T): T
@type = T => (T): T

// 运行函数得到一个类型 F,即类型为 (number): number
@F = @type(number)

// 要求 increase 这个函数符合 F 这种类型,也就是参数为 number,返回值也为 number
@@F
function increase(param) {
 // ...
}
@@end

实际上没有 @@F 这种语法,是我编造出来的,目的是让你可以从另一个角度去看类型系统。

当我们理解泛型是一种“参数”之后,我们可能会问:类型系统的函数在哪里?对于 js 函数而言,你可以很容易指出函数声明语句和参数,但是 ts 中,这个部分是隐藏起来的。不过,我们可以在一些特定结构中,比较容易看到类型函数的影子:


// 声明一个泛型接口,这个写法,像极了声明一个函数,我们用描述语言来形容 @type = T => (T): T
interface GenericIdentityFn<T> {
   (arg: T): T;
}

// 这个写法,有点像一个闭包函数,在声明函数后,立即运行这个函数,描述语言:@@[T => (T): T](any)
function identity<T>(arg: T): T {
   return arg;
}

// 使用泛型接口,像极了调用一个函数,我们用描述语言来形容 @type(number)
let myIdentity: GenericIdentityFn<number> = identity;

上面这一整段代码,我们用描述文本重写一遍:


@GenericIdentityFn = T => (T): T

@@[T => (T): T](any)
function identify(arg) {
 return arg
}
@@end

@@GenericIdentityFn(number)
let myIdentity = identity
@@end

我们在类型系统中声明了两个函数,分别是 @GenericIdentityFn 和 @some(匿名函数 @[T => (T): T])。虽然是两个函数,但是实际上,它们的是一模一样的,因为 typescript 是结构类型,也就是在类型检查的时候只判断结构上的每个节点类型是否相同,而不是必须保持类型变量本身的指针相同。@GenericIdentityFn 和 @some 这两个函数分别被调用,用来修饰 identify 和 myIdentify,在调用的时候,接收的参数不同,所以导致最终的类型检查规则是不同的,identify 只要保证参数和返回值的类型相同,至于具体什么类型,any。而 myIdentify 除了保证参数返回值类型相同外,还要求类型必须是 number。

泛型类

除了泛型接口,class 类也可以泛型化,即“泛型类”,借助泛型类,我们来探究一下泛型的声明和使用的步骤。


class GenericNumber<T> {
   zeroValue: T;
   add: (x: T, y: T) => T;
}

let myGenericNumber = new GenericNumber<number>();

前文泛型接口因为只是为了约束函数的类型,所以写的很像函数,实际上,我们可以用描述语言重新描述一个泛型接口和泛型类。上面的红色部分,我们用描述语言来描述:


@GenericNumber = T => class {
 zeroValue: T;
 add: (x: T, y: T) => T;
}

@GenericNumber 这个函数,以 T 为参数,返回一个 class,在 @type 函数体内多次用到了参数 T。


@GenericIdentityFn = T => interface {
 (arg: T): T;
}

我们重新描述了前面的 interface GenericIdentityFn,这样我们就可以在接口中增加其他的方法。

可以注意到,即使 typescript 内置的基础类型,例如 Array,被声明为泛型接口、泛型类之后,这些接口和类在使用时必须通过<>传入参数,本质上,因为它们都是函数,只是返回值不同。

其他泛型使用的通俗解释

接下来我们要再描述一个复杂的类型:


class Animal {
   numLegs: number;
}

function createInstance<A extends Animal>(c: new () => A): A {
   return new c();
}

我们姑且不去看 new() 的部分,我们看尖括号中的 extends语法,这里应该怎么理解呢?实际上,我们面对的问题是,在编译时,<A extends Animal> 尖括号中的内容是什么时候运行的,是之前,还是之间?


// 到底是
@type = (A extends Animal) => (new() => A): A
@type(T)
// 还是
@type = A => (new() => A): A
@type(T extends Animal)复

因为 typescript 是静态类型系统,Animal 是不变的类,因此,可以推测其实在类的创建之前,尖括号的内容已经被运行了。


@type = (A extends Animal) => (new() => A): A

也就是说,要使用 @type(T) 产生类型,首先 T 要满足 Animal 的结构,然后才能得到需要的类型,如果 T 已经不满足 Animal 类的结构了,那么编译器会直接报错,而这个报错,不是类型检查阶段,而是在类型系统的创建阶段,也就是 ts 代码的运行阶段。这种情况被称为“泛型约束”。

另外,类似 <A,B> 这样的语法其实和函数参数一致。


@type = (A, B) => (A|B): SomeType

我们再来看 ts 内置的基础类型:Array<number>


@Array = any => any[]

来源:https://www.cnblogs.com/qianxiaox/p/14054540.html

标签:ts,泛型
0
投稿

猜你喜欢

  • Python制作简单的网页爬虫

    2022-10-09 12:49:16
  • Django零基础入门之运行Django版的hello world

    2022-11-10 02:44:58
  • python简单实现刷新智联简历

    2023-09-08 06:49:44
  • python类中的self和变量用法及说明

    2022-05-27 10:33:12
  • Windows系统下mysql5.7.21安装详细教程

    2024-01-21 17:26:38
  • MQTT.js 入门使用教程

    2024-04-19 09:57:20
  • 使用Perl语言去存取mSQL和MySQL数据库的内容

    2009-10-23 09:11:00
  • python3实现语音转文字(语音识别)和文字转语音(语音合成)

    2022-10-02 03:40:33
  • 老版本PHP转义Json里的特殊字符的函数

    2023-11-06 02:32:18
  • 富文本编辑器的基本原理与实践

    2008-06-13 13:28:00
  • C#连接Oracle数据库的方法

    2024-01-16 20:17:12
  • Mysql导入导出工具Mysqldump和Source命令用法详解

    2024-01-23 19:46:52
  • numba提升python运行速度的实例方法

    2022-10-06 20:39:27
  • 教你用python控制安卓手机

    2023-11-10 10:13:39
  • SQL Server实现全文搜索查询详解

    2024-01-23 05:19:13
  • python如何实现MK突变检验方法,代码复制修改可用

    2022-04-10 13:31:18
  • js判断手机和pc端选择不同执行事件的方法

    2024-04-29 13:45:46
  • python中enumerate函数遍历元素用法分析

    2021-08-07 10:07:18
  • Python3爬虫关于识别检验滑动验证码的实例

    2021-08-25 02:17:37
  • Python基于Opencv来快速实现人脸识别过程详解(完整版)

    2023-09-06 17:04:25
  • asp之家 网络编程 m.aspxhome.com