我的编程空间,编程开发者的网络收藏夹
学习永远不晚

如何通俗的解释TypeScript 泛型

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

如何通俗的解释TypeScript 泛型

概述

在 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[]

结语

Typescript 中的泛型,实际上就是类型的生成函数的参数。本文的内容全部为凭空想象,仅适用于对 ts 进行理解时的思路开拓,不适用于真实编程,特此声明。

以上就是如何通俗的解释TypeScript 泛型的详细内容,更多关于TypeScript泛型的资料请关注编程网其它相关文章!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

如何通俗的解释TypeScript 泛型

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Go语言泛型解读:它的泛型特性到底如何?

Go语言是一种快速、简单、安全的编程语言,被广泛应用于各种领域。在过去的版本中,Go语言一直以“没有泛型”而著称,但是随着Go 1.18版本的发布,引入了长期期待的泛型特性,这是Go语言发展历史上的一大里程碑。本文将针对Go语言的泛型特性进
Go语言泛型解读:它的泛型特性到底如何?
2024-03-15

如何使用泛型编写更通用的golang代码

泛型特性允许 go 语言编写可应用于多种数据类型的通用代码。它通过使用泛型类型变量实现,例如 type mytype[t any],其中 t 指定元素类型。泛型提供了以下优势:代码可重用:通用代码可同时适用于各种类型,无需编写特定类型代码。
如何使用泛型编写更通用的golang代码
2024-05-03

如何理解 SFINAE 在 C++ 泛型编程中的作用?

sfinae 允许函数模板根据参数类型判断,在泛型编程中对条件检查非常有用。它通过添加返回 void 的参数实现:如果传入类型有效,则不会报错。如果传入类型无效,则实例化函数模板会失败,因为编译器不知道如何处理 void 参数。实战案例中,
如何理解 SFINAE 在 C++ 泛型编程中的作用?
2024-04-24

Golang泛型如何解决函数中与类型相关的错误?

泛型解决了 go 函数中与类型相关的错误,允许创建通用函数,可接受并操作各种类型的数据。通过使用类型变量,泛型函数可以适应传递给它们的任何类型,从而避免类型冲突错误,并简化和提高代码的灵活性。Go 泛型用于解决函数中与类型相关的错误在 G
Golang泛型如何解决函数中与类型相关的错误?
2024-04-16

如何理解Java中的泛型?泛型有哪些应用场景?(Java泛型概念是什么?它在哪些场合下特别有用?)

Java泛型允许创建类型独立的类和方法,使用类型参数表示抽象类型,从而消除原始类型和强制转换的需要。泛型广泛应用于:代码复用:创建可重用的代码,支持不同类型。类型安全:强制编译时类型检查,消除运行时类型错误。性能提升:避免原始类型和强制转换,提高代码性能。具体应用场景包括集合框架(如List)、算法(如类型独立的排序算法)、数据结构和泛型方法。通配符允许放松泛型类型限制,增加代码灵活性。
如何理解Java中的泛型?泛型有哪些应用场景?(Java泛型概念是什么?它在哪些场合下特别有用?)
2024-04-02

PyTorch中如何进行模型的解释性分析

PyTorch提供了多种方法来进行模型的解释性分析,以下是一些常用的方法:特征重要性分析:可以使用SHAP(SHapley Additive exPlanations)库来计算特征的重要性,帮助理解模型的预测结果是如何基于输入特征而变化的。
PyTorch中如何进行模型的解释性分析
2024-03-05

泛型的类型擦除后fastjson反序列化时如何还原详解

这篇文章主要为大家介绍了泛型的类型擦除后fastjson反序列化时如何还原详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2022-11-13

Java高级特性中的泛型、反射和注解该如何理解

这篇文章跟大家分析一下“Java高级特性中的泛型、反射和注解该如何理解”。内容详细易懂,对“Java高级特性中的泛型、反射和注解该如何理解”感兴趣的朋友可以跟着小编的思路慢慢深入来阅读一下,希望阅读后能够对大家有所帮助。下面跟着小编一起深入
2023-06-16

如何解决MyBatis在DAO层定义接口返回类型泛型无效的问题

这篇文章主要讲解了“如何解决MyBatis在DAO层定义接口返回类型泛型无效的问题”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“如何解决MyBatis在DAO层定义接口返回类型泛型无效的问题
2023-06-20

如何解决mybatis中方法返回泛型与resultType不一致的问题

这篇文章主要介绍“如何解决mybatis中方法返回泛型与resultType不一致的问题”,在日常操作中,相信很多人在如何解决mybatis中方法返回泛型与resultType不一致的问题问题上存在疑惑,小编查阅了各式资料,整理出简单好用的
2023-06-20

编程热搜

目录