typescript中的泛型

一、泛型的基本概念

泛型就像是一个"类型占位符",它允许我们在定义函数、类或接口时暂不指定具体类型,等到使用时再确定具体类型。这就像做月饼的模具,模具的形状固定(泛型的约束),但可以填充不同口味的馅料(具体类型)。

在你这段代码中的 <T> 就是泛型参数,读作"类型T"。


二、为什么要用泛型?

观察你的函数定义:

typescript 复制代码
function echoWithLength<T extends IWithLength>(arg: T): T {
  console.log(arg.length);
  return arg;
}

如果不用泛型,代码可能会写成:

typescript 复制代码
function echoWithLength(arg: IWithLength): IWithLength {
  console.log(arg.length);
  return arg;
}

这两种写法的关键区别在于类型信息的保留

调用示例 泛型版本返回类型 非泛型版本返回类型
echoWithLength("hello") string IWithLength
echoWithLength([1,2,3]) number[] IWithLength

使用泛型时,返回类型会与输入类型完全一致,保留了更多类型信息,这是泛型的核心优势。


三、extends 的作用:类型约束

<T extends IWithLength> 的语法表示: "类型T必须是实现了IWithLength接口的类型,即必须有length: number属性"

这就像给泛型T加上了一个安全限制:

typescript 复制代码
interface IWithLength {
  length: number; // 必须满足这个条件
}

实际效果: ✅ 允许的类型:string, any[], {length: number} ❌ 不允许的类型:number, boolean, {}(没有length属性的对象)


四、你的代码执行过程分析

1. 字符串调用

typescript 复制代码
const str1 = echoWithLength('arr');
  • T 被推断为 string 类型
  • 字符串天然具有 length 属性
  • 返回类型仍是 string

2. 对象调用

typescript 复制代码
const obj = echoWithLength({ length: 10 });
  • T 被推断为 { length: number }
  • 显式提供了 length 属性
  • 返回类型保持为 { length: number }

3. 数组调用

typescript 复制代码
const arr2 = echoWithLength([1,2,3]);
  • T 被推断为 number[]
  • 数组天然具有 length 属性
  • 返回类型仍是 number[]

五、类型推断的智能表现

TypeScript 编译器会自动进行类型推断,你甚至可以不显式指定泛型:

typescript 复制代码
// 编译器会自动推断 T 为 string
echoWithLength("hello"); 

// 等同于显式声明
echoWithLength<string>("hello");

六、为什么不能直接用 any?

如果代码写成:

typescript 复制代码
function echoWithLength(arg: any): any {
  console.log(arg.length);
  return arg;
}

这样会失去:

  1. 参数的类型检查
  2. 返回值的类型关联
  3. 自动补全等IDE支持

七、实际应用场景

这种模式常见于需要保持输入输出类型一致,同时需要访问某些公共属性的场景:

  1. 处理集合(数组、字符串等)的工具函数
  2. 需要 length 属性的数据验证
  3. React组件 props 的类型约束

八、学习建议

要深入理解泛型,推荐尝试以下练习:

  1. 写一个返回数组最后一个元素的函数
  2. 实现一个支持多种类型的键值对存储类
  3. 创建一个能合并两个对象并保留类型信息的函数

简单说一下ts中的泛型,它是一个类型的占位符,可以用在函数类接口中的定义当中,可以先不指定具体类型,等到时候使用时才确定具体类型。

相关推荐
GISer_Jing7 小时前
阿里开源纯前端浏览器自动化 PageAgent,[特殊字符] 浏览器自动化变天啦?
前端·人工智能·自动化·aigc·交互
清风徐来QCQ7 小时前
js中的模板字符串
开发语言·前端·javascript
成都渲染101云渲染66667 小时前
Houdini+Blender高效渲染方案(高配算力+全渲染器兼容)
前端·系统架构
奕成则成7 小时前
面试被问:MySQL 与 Doris/SelectDB 的架构区别。 大数据为什么禁止select *。
mysql·面试·架构
SuperEugene7 小时前
Vue3 + Element Plus 表格实战:批量操作、行内编辑、跨页选中逻辑统一|表单与表格规范篇
开发语言·前端·javascript
AlunYegeer7 小时前
面试问题controller和service能不能互相替换
面试·职场和发展
MSTcheng.7 小时前
【优选算法必修篇——位运算】『面试题 01.01. 判定字符是否唯一&面试题 17.19. 消失的两个数字』
java·算法·面试
极梦网络无忧8 小时前
基于 Vite + Vue3 的组件自动注册功能
前端·javascript·vue.js
Predestination王瀞潞8 小时前
5.4.3 通信->WWW万维网内容访问标准(W3C):WWW(World Wide Web) 协议架构(分层)
前端·网络·网络协议·架构·www
爱学习的程序媛8 小时前
【Web前端】优化Core Web Vitals提升用户体验
前端·ui·web·ux·用户体验