TypeScript基础(五)泛型

✨ 专栏介绍

TypeScript是一种由微软开发的开源编程语言,它是JavaScript的超集,意味着任何有效的JavaScript代码都是有效的TypeScript代码。TypeScript通过添加静态类型和其他特性来增强JavaScript,使其更适合大型项目和团队开发。

在TypeScript专栏中,我们将深入探讨TypeScript的各个方面,包括语法、类型系统、模块化、面向对象编程等。我们将介绍如何使用TypeScript来构建可维护、可扩展和高效的应用程序。

TypeScript是一种开源的编程语言,它是JavaScript的超集,意味着所有的JavaScript代码都可以在TypeScript中运行。TypeScript添加了静态类型检查和其他一些新特性,以提高代码的可读性、可维护性和可扩展性。
在这里插入图片描述

文章目录

    • ✨ 专栏介绍
    • 引言
    • 概念
    • 泛型的使用
        • 1. 函数中使用泛型
        • 2. 类型别名中使用泛型
        • 3. 接口中使用泛型
        • 4. 类中使用泛型
    • 泛型约束与多泛型
        • 1. 泛型约束
        • 2. 多泛型
    • 示例
    • 总结
    • 😶 写在结尾


引言

在编程中,我们经常会遇到需要处理不同类型数据的情况。为了提高代码的复用性和灵活性,TypeScript引入了泛型的概念。泛型可以让我们在定义函数、类或接口时,不预先指定具体的类型,而是在使用时再指定类型。本文将详细介绍TypeScript中泛型的使用方法和技巧。

概念

泛型是一种参数化类型的方式,它可以用来创建可重用的组件。通过使用泛型,我们可以在定义函数、类或接口时不预先指定具体的类型,而是在使用时再指定类型。这样可以增加代码的灵活性和复用性。

泛型的使用

在函数、类型别名、接口和类中使用泛型可以增加代码的灵活性和重用性。下面详细介绍如何在这些场景中使用泛型,并提供相应的示例。

1. 函数中使用泛型

函数可以使用泛型来接收不同类型的参数,并返回相应的结果。可以通过在函数名后面使用尖括号(<>)来定义泛型参数,并在函数体内使用该参数。

function identity<T>(arg: T): T {
  return arg;
}

let result = identity<string>("Hello");
console.log(result);  // 输出:Hello

let result2 = identity<number>(123);
console.log(result2);  // 输出:123
2. 类型别名中使用泛型

类型别名可以用来定义复杂的类型,包括泛型类型。可以通过在类型别名后面使用尖括号(<>)来定义泛型参数,并在类型定义中使用该参数。

type Pair<T> = {
  first: T;
  second: T;
};

let pair: Pair<number> = { first: 1, second: 2 };
console.log(pair);  // 输出:{ first: 1, second: 2 }

let pair2: Pair<string> = { first: "hello", second: "world" };
console.log(pair2);  // 输出:{ first: 'hello', second: 'world' }
3. 接口中使用泛型

接口可以使用泛型来定义灵活的类型。可以通过在接口名后面使用尖括号(<>)来定义泛型参数,并在接口定义中使用该参数。

interface Box<T> {
  value: T;
}

let box: Box<number> = { value: 123 };
console.log(box);  // 输出:{ value: 123 }

let box2: Box<string> = { value: "hello" };
console.log(box2);  // 输出:{ value: 'hello' }
4. 类中使用泛型

类可以使用泛型来定义灵活的属性和方法。可以通过在类名后面使用尖括号(<>)来定义泛型参数,并在类定义中使用该参数。

class Queue<T> {
  private elements: T[] = [];

  enqueue(element: T): void {
    this.elements.push(element);
  }

  dequeue(): T | undefined {
    return this.elements.shift();
  }

  isEmpty(): boolean {
    return this.elements.length === 0;
  }

  size(): number {
    return this.elements.length;
  }
}

let queue = new Queue<number>();

queue.enqueue(1);
queue.enqueue(2);

console.log(queue.dequeue());  // 输出:1
console.log(queue.size());  // 输出:1

泛型约束与多泛型

泛型约束和多泛型是在使用泛型时的一些高级技巧,可以进一步限制泛型的类型范围和增加灵活性。下面详细介绍泛型约束和多泛型,并提供相应的示例说明。

1. 泛型约束

泛型约束可以限制泛型参数必须满足某些条件,例如必须是某个基类的子类、必须实现某个接口等。可以使用 extends 关键字来定义泛型约束。

interface Lengthwise {
  length: number;
}

function loggingIdentity<T extends Lengthwise>(arg: T): T {
  console.log(arg.length);
  return arg;
}

loggingIdentity("hello"); // 输出:5
loggingIdentity([1, 2, 3]); // 输出:3
loggingIdentity({ length: 10, value: 3 }); // 输出:10
2. 多泛型

可以同时定义多个泛型参数,用逗号分隔。多个泛型参数可以相互之间有关联,也可以完全独立。

function merge<T, U>(obj1: T, obj2: U): T & U {
  return { ...obj1, ...obj2 };
}

let mergedObj = merge({ name: "John" }, { age: 30 });

console.log(mergedObj.name); // 输出:John
console.log(mergedObj.age); // 输出:30

在上面的示例中,merge 函数接收两个参数,一个是类型为 T 的对象 obj1,另一个是类型为 U 的对象 obj2。函数返回的类型是 T & U,表示返回的对象同时具有 TU 类型的属性。

需要注意以下几点:

  • 泛型约束使用 extends 关键字来定义,可以约束泛型参数必须满足某些条件。

  • 泛型约束可以应用于泛型函数、泛型类和泛型接口。

  • 多个泛型参数可以相互之间有关联,也可以完全独立。

  • 在使用多泛型时,需要注意传入的参数类型和返回值类型要与泛型参数相匹配,否则可能会导致编译错误或运行时错误。

示例

开发一个字典类(Dictionary),字典中会保存键值对的数据

键值对数据的特点:

  • 键(key)可以是任何类型,但不允许重复
  • 值(value)可以是任何类型
  • 每个键对应一个值
  • 所有的键类型相同,所有的值类型相同
export type CallBack<T, U> = (key: T, val: U) => void

export class Dictionary<K, V> {
  private keys: K[] = []
  private values: V[] = []

  get size() {
    return this.keys.length
  }

  set(key: K, val: V) {
    const i = this.keys.indexOf(key)
    if (i < 0) {
      this.keys.push(key)
      this.values.push(val)
    } else {
      this.values[i] = val
    }
  }

  forEach(callback: CallBack<K, V>) {
    this.keys.forEach((k, i) => {
      const v = this.values[i]
      callback(k, v)
    })
  }

  has(key: K) {
    return this.keys.includes(key)
  }

  delete(key: K) {
    const i = this.keys.indexOf(key)
    if (i === -1) {
      return
    }
    this.keys.splice(i, 1)
    this.values.splice(i, 1)
  }
}

这个泛型类和泛型类型别名可以实现一个通用的字典数据结构。可以根据需要传入不同类型的键和值来创建字典对象,并使用提供的方法进行操作。例如:

const dict = new Dictionary<string, number>();

dict.set("one", 1);
dict.set("two", 2);
dict.set("three", 3);

console.log(dict.size); // 输出:3

dict.forEach((key, value) => {
  console.log(key, value);
});
// 输出:
// one 1
// two 2
// three 3

console.log(dict.has("two")); // 输出:true

dict.delete("two");
console.log(dict.has("two")); // 输出:false

需要注意的是,在使用泛型类和泛型类型别名时,可以根据实际需求传入不同的类型参数,以适应不同的数据类型。

总结

泛型是TypeScript中非常重要的特性之一,它可以让我们在定义函数、类或接口时不预先指定具体的类型,而是在使用时再指定类型。通过使用泛型,我们可以增加代码的灵活性和复用性。在函数中使用泛型时,可以通过传入具体的类型参数来调用函数。在类型别名、接口、类中使用泛型时,可以在定义时指定类型参数,并在使用时传入具体的类型。同时,我们还可以对泛型进行约束以确保传入的类型满足某些条件。

使用泛型的一些技巧和需要注意的点如下:

  • 可以同时定义多个泛型参数,例如 function foo<T, U>(arg1: T, arg2: U): void { ... }
  • 可以在泛型参数上使用约束,例如 function foo<T extends SomeType>(arg: T): void { ... },其中 SomeType 是一个已知的类型。
  • 可以在泛型参数上使用默认类型,例如 function foo<T = SomeType>(arg: T): void { ... },其中 SomeType 是一个已知的类型。
  • 在使用泛型时,可以显式指定泛型参数的类型,也可以让编译器自动推断泛型参数的类型。
  • 在使用泛型时,需要注意传入的参数类型和返回值类型要与泛型参数相匹配,否则可能会导致编译错误或运行时错误。

😶 写在结尾

前端设计模式专栏
在这里插入图片描述
设计模式是软件开发中不可或缺的一部分,它们帮助我们解决了许多常见问题,并提供了一种优雅而可靠的方式来构建应用程序。在本专栏中,我们介绍了所有的前端设计模式,包括观察者模式、单例模式、策略模式等等。通过学习这些设计模式,并将其应用于实际项目中,我们可以提高代码的可维护性、可扩展性和可重用性。希望这个专栏能够帮助你在前端开发中更好地应用设计模式,写出高质量的代码。点击订阅前端设计模式专栏

Vue专栏
在这里插入图片描述
Vue.js是一款流行的JavaScript框架,用于构建用户界面。它采用了MVVM(Model-View-ViewModel)的架构模式,通过数据驱动和组件化的方式,使开发者能够更轻松地构建交互性强、可复用的Web应用程序。在这个专栏中,我们将深入探讨Vue.js的核心概念、组件开发、状态管理、路由和性能优化等方面的知识。我们将学习如何使用Vue.js构建响应式的用户界面,并探索其强大的生态系统,如Vue Router和Vuex、Pinia。通过学习这些内容,你将能够成为一名熟练的Vue.js开发者,并能够应用这些知识来构建复杂而高效的Web应用程序。点击订阅Vue专栏

JavaScript(ES6)专栏在这里插入图片描述
JavaScript是一种广泛应用于网页开发和后端开发的脚本语言。它具有动态性、灵活性和易学性的特点,是构建现代Web应用程序的重要工具之一。在这个专栏中,我们将深入探讨JavaScript语言的基本语法、DOM操作、事件处理、异步编程以及常见算法和数据结构等内容。此外,我们还将介绍ES6(ECMAScript 2015)及其后续版本中引入的新特性,如箭头函数、模块化、解构赋值等。通过学习这些内容,你将能够成为一名熟练的JavaScript开发者,并能够应用这些知识来构建出高质量和可维护的Web应用程序。点击订阅JavaScript(ES6)专栏

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/306609.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【机器学习】模型参数优化工具:Optuna使用分步指南(附XGB/LGBM调优代码)

常用的调参方式和工具包 常用的调参方式包括网格搜索(Grid Search)、**随机搜索(Random Search)和贝叶斯优化(Bayesian Optimization)**等。 工具包方面&#xff0c;Scikit-learn提供了GridSearchCV和RandomizedSearchCV等用于网格搜索和随机搜索的工具。另外&#xff0c;有一…

CodeWave智能开发平台--03--目标:应用创建--08联系人管理

摘要 本文是网易数帆CodeWave智能开发平台系列的第11篇&#xff0c;主要介绍了基于CodeWave平台文档的新手入门进行学习&#xff0c;实现一个完整的应用&#xff0c;本文主要完成08联系人管理 CodeWave智能开发平台的11次接触 CodeWave参考资源 网易数帆CodeWave开发者社区…

Linux第21步_取消鼠标中键的复制粘贴功能

在ubuntu18.04操作系统中&#xff0c;选中文本后&#xff0c;若按下鼠标中键&#xff0c;就可以执行复制粘贴&#xff0c;相当于 CtrlshiftC 后又按了 CtrlshiftV。在Linux系统中&#xff0c;基本上都是这么配置的。在windows系统中&#xff0c;我们习惯用Ctrl-C复制&#xff0…

intellij idea导入别人项目版本问题解决方案

当导入别人的项目太慢,原因是gradle版本不一致,这时android studio自动下载匹配的gradle版本导致长时间下载的问题。原因主要还是&#xff1a;这个下载地址是国外的&#xff0c;需要翻墙&#xff0c;否则会特别慢。 1.一般下载下来的项目都有这些文件夹&#xff0c;在导入项目…

51单片机介绍

1 单片机简介 单片机&#xff0c;英文Micro Controller Unit&#xff0c;简称MCU 内部集成了CPU、RAM、ROM、定时器、中断系统、通讯接口等一系列电脑的常用硬件功能 单片机的任务是信息采集&#xff08;依靠传感器&#xff09;、处理&#xff08;依靠CPU&#xff09;和硬件设…

uniapp 创建组件

组件&#xff1a;用于将某个功能的 HTML、CSS、JS 封装到一个文件中&#xff0c;提高代码的复用性和可维护性。 创建组件 一、在根目录中创建 components 文件夹&#xff0c;右键点击新建组件。 二、输入组件名称、选择默认模板、点击创建组件。 三、在组件中正常编写内容即可…

AcWing 203. 同余方程(扩展欧几里得算法)

题目链接 203. 同余方程 - AcWing题库高质量的算法题库https://www.acwing.com/problem/content/205/ 来源 《算法竞赛进阶指南》, NOIP2012提高组 题解 本题中的同余方程可以转化为ax by 1的形式&#xff0c;利用扩展欧几里得算法可以求得特解为&#xff0c;则通解为。 代…

Linux系统使用超详细(九)~用户和组管理

本篇将要梳理有关用户和用户组的学习笔记&#xff0c;内容主要是基本的概念理解和常用命令的使用方法 &#xff01; 目录 一、用户和用户组认识 1.1用户说明 1.1.1查看用户信息 ①id命令&#xff1a; ②whoami 命令 ③cat /etc/passwd 命令 ④getent passwd 命令 ⑤仅显…

Python轴承故障诊断 (十)基于VMD+CNN-Transfromer的故障分类

目录 1 变分模态分解VMD的Python示例 2 轴承故障数据的预处理 2.1 导入数据 2.2 故障VMD分解可视化 3 基于VMDCNN-Transformer的轴承故障诊断分类 3.1 定义VMD-CNN-Transformer分类网络模型 3.2 设置参数&#xff0c;训练模型 3.3 模型评估 代码、数据如下&#xff1a…

​如何在iOS手机上查看应用日志

引言 在开发iOS应用过程中&#xff0c;查看应用日志是非常重要的一项工作。通过查看日志&#xff0c;我们可以了解应用程序运行时的状态和错误信息&#xff0c;帮助我们进行调试和排查问题。本文将介绍两种方法来查看iOS手机上的应用日志&#xff0c;并提供相应的操作步骤。 …

vercel部署twikoo后评论收不到通知邮件问题解决方法

&#x1f4cc; 前言&#xff1a;本文主要是总结一下在vercel部署twikoo后收不到评论邮件通知问题的解决方法&#xff0c;本人在各种查资料无果后最终去twioo的git官方项目的issue中找到某位大佬给出的原因以及解决方案&#xff0c;故做此记录&#xff0c;希望对遇到此问题的同学…

Nodejs 第三十一章(响应头和请求头)

响应头 HTTP响应头&#xff08;HTTP response headers&#xff09;是在HTTP响应中发送的元数据信息&#xff0c;用于描述响应的特性、内容和行为。它们以键值对的形式出现&#xff0c;每个键值对由一个标头字段&#xff08;header field&#xff09;和一个相应的值组成。 例如…

C++面试宝典第17题:找规律填数

题目 仔细观察下面的数字序列,找到规律,并填写空白处的数字。 (1)1, 2, 4, 7, 11, 16, __ (2)-1, 2, 7, 28, __, 126 (3)6, 10, 18, 32, 57, __ (4)19, 6, 1, 2, 11, __ (5)2, 3, 5, 7, 11, __ (6)1, 8, 9, 4, __, 1/6 (7)1, 2, 3, 7, 16, __, 321 (8)1, 2, …

【Python学习】Python学习9-字符串

目录 【Python学习】Python学习9-字符串 前言创建语法访问字符串的值字符串拼接Python 转义字符Python字符串运算符Python格式化字符串Python 三引号Unicode字符串Python 的字符串内建函数参考 文章所属专区 Python学习 前言 本章节主要说明Python的字符串类型。 创建语法 …

Spring MVC组件

1.DispatcherServlet前端控制器 用户请求到达前端控制器&#xff0c;它就相当于mvc模式中的c&#xff0c;dispatcherServlet 是整个流程控制的中心&#xff0c;由它调用其它组件处理用户的请求&#xff0c;dispatcherServlet 的存在降低了组件之间的耦合性。 2.HandlerMappin…

蓝牙物联网多个核心应用场景开发与应用细化分析

蓝牙物联网是指利用蓝牙技术将物理设备与互联网连接起来&#xff0c;实现设备之间的信息共享与互通。蓝牙物联网在各个领域得到了广泛应用&#xff0c;并且在未来有着巨大的发展潜力。本文将围绕蓝牙物联网的五大核心应用场景进行介绍&#xff0c;包括智能家居、智能健康、智能…

Mac M1 Parallels CentOS7.9 Install Jenkins

官网: https://www.jenkins.io/ 文中涉及的jenkins’s rpm链接: https://pan.baidu.com/s/1BSe62pGdJCtDUAimaniEMA?pwd6666 一、Install & Check Java Env Oracle官网下载Java: https://www.oracle.com/cn/ # 拷贝到Jenkins服务器 scp Downloads/jdk-8u391-linux-aarch…

再谈前端算法

楔子 – 青蛙跳台阶什么是算法算法实例 &#xff1a; 实现一个LRU缓存 实现 LRUCache扩展&#xff1a; ES6 Map Map的创建和初始化&#xff1a;添加键值对&#xff1a;获取键值对&#xff1a;检查Map中是否存在某个键&#xff1a;删除键值对&#xff1a;遍历Map&#xff1a;获取…

读算法霸权笔记13_读后总结与感想兼导读

1. 基本信息 算法霸权&#xff1a;数学杀伤性武器的威胁 [美] 凯西奥尼尔(Cathy 著 中信出版社,2018年9月出版 1.1. 读薄率 书籍总字数220千字&#xff0c;笔记总字数32359字。 读薄率32359220000≈14.71% 1.2. 读厚方向 算法的力量&#xff1a;人类如何共同生存&#x…

SpringBoot-开启Admin监控服务

SpringBoot-Admin是一个用于管理和监控SpringBoot应用程序的开源项目。它提供了一个易于使用的Web界面&#xff0c;可以实时监控应用程序的健康状况、性能指标、日志和环境配置等信息。通过Actuator模块来收集和暴露应用程序的监控信息&#xff0c;使用Web Socket或者Server-Se…