文章详情

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

请输入下面的图形验证码

提交验证

短信预约提醒成功

面试官:说说你对 TypeScript 中泛型的理解?应用场景?

2024-12-14 00:42

关注

一、是什么

泛型程序设计(generic programming)是程序设计语言的一种风格或范式

泛型允许我们在强类型程序设计语言中编写代码时使用一些以后才指定的类型,在实例化时作为参数指明这些类型 在typescript中,定义函数,接口或者类的时候,不预先定义好具体的类型,而在使用的时候在指定类型的一种特性

假设我们用一个函数,它可接受一个 number 参数并返回一个number 参数,如下写法:

  1. function returnItem (para: number): number { 
  2.     return para 

如果我们打算接受一个 string 类型,然后再返回 string类型,则如下写法:

  1. function returnItem (para: string): string { 
  2.     return para 

上述两种编写方式,存在一个最明显的问题在于,代码重复度比较高

虽然可以使用 any类型去替代,但这也并不是很好的方案,因为我们的目的是接收什么类型的参数返回什么类型的参数,即在运行时传入参数我们才能确定类型

这种情况就可以使用泛型,如下所示:

  1. function returnItem(para: T): T { 
  2.     return para 

可以看到,泛型给予开发者创造灵活、可重用代码的能力

二、使用方式

泛型通过<>的形式进行表述,可以声明:

函数声明

声明函数的形式如下:

  1. function returnItem(para: T): T { 
  2.     return para 

定义泛型的时候,可以一次定义「多个类型参数」,比如我们可以同时定义泛型 T 和 泛型 U:

  1. function swap(tuple: [T, U]): [U, T] { 
  2.     return [tuple[1], tuple[0]]; 
  3.  
  4. swap([7, 'seven']); // ['seven', 7] 

接口声明

声明接口的形式如下:

  1. interface ReturnItemFn { 
  2.     (para: T): T 

那么当我们想传入一个number作为参数的时候,就可以这样声明函数:

  1. const returnItem: ReturnItemFn = para => para 

类声明

使用泛型声明类的时候,既可以作用于类本身,也可以作用于类的成员函数

下面简单实现一个元素同类型的栈结构,如下所示:

  1. class Stack { 
  2.     private arr: T[] = [] 
  3.  
  4.     public push(item: T) { 
  5.         this.arr.push(item) 
  6.     } 
  7.  
  8.     public pop() { 
  9.         this.arr.pop() 
  10.     } 

使用方式如下:

  1. const stack = new Stacn() 

如果上述只能传递 string 和 number 类型,这时候就可以使用 的方式猜实现「约束泛型」,如下所示:

除了上述的形式,泛型更高级的使用如下:

例如要设计一个函数,这个函数接受两个参数,一个参数为对象,另一个参数为对象上的属性,我们通过这两个参数返回这个属性的值

这时候就设计到泛型的索引类型和约束类型共同实现

索引类型、约束类型

索引类型 keyof T 把传入的对象的属性类型取出生成一个联合类型,这里的泛型 U 被约束在这个联合类型中,如下所示:

  1. function getValue(obj: T, key: U) { 
  2.   return obj[key] // ok 

上述为什么需要使用泛型约束,而不是直接定义第一个参数为 object类型,是因为默认情况 object 指的是{},而我们接收的对象是各种各样的,一个泛型来表示传入的对象类型,比如 T extends object

使用如下图所示:

多类型约束

例如如下需要实现两个接口的类型约束:

  1. interface FirstInterface { 
  2.   doSomething(): number 
  3.  
  4. interface SecondInterface { 
  5.   doSomethingElse(): string 

可以创建一个接口继承上述两个接口,如下:

  1. interface ChildInterface extends FirstInterface, SecondInterface { 
  2.  

正确使用如下:

  1. class Demo { 
  2.   private genericProperty: T 
  3.  
  4.   constructor(genericProperty: T) { 
  5.     this.genericProperty = genericProperty 
  6.   } 
  7.   useT() { 
  8.     this.genericProperty.doSomething() 
  9.     this.genericProperty.doSomethingElse() 
  10.   } 

通过泛型约束就可以达到多类型约束的目的

三、应用场景

通过上面初步的了解,后述在编写 typescript 的时候,定义函数,接口或者类的时候,不预先定义好具体的类型,而在使用的时候在指定类型的一种特性的时候,这种情况下就可以使用泛型

灵活的使用泛型定义类型,是掌握typescript 必经之路

参考文献

 

https://www.tslang.cn/docs/handbook/generics.html

 

来源:JS每日一题内容投诉

免责声明:

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

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

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯