解锁 TypeScript 的元编程魔法:从 `extends` 到 `infer` 的条件类型之旅

一、类型的三元表达式

条件类型的语法看起来就像 JavaScript 的三元运算符:

typescript 复制代码
SomeType extends OtherType ? TrueType : FalseType;

解析:

  • SomeType extends OtherType:这是我们的"条件判断"。这里的 extends 关键字不是类的继承,而是泛型里面的条件限定判断
  • ? TrueType:如果条件为真,那么最终的类型就是 TrueType
  • : FalseType:如果条件为假,那么最终的类型就是 FalseType

示例:

typescript 复制代码
// IsNumber<T> 会检查类型 T 是否是 number
type IsNumber<T> = T extends number ? "yes, it's a number" : 'no, not a number';

type Result1 = IsNumber<100>; // "yes, it's a number"
type Result2 = IsNumber<'hello'>; // "no, not a number"

let t: Result1 = "yes, it's a number"; // 只能是这个值
let b: Result2 = 'no, not a number';// 只能是这个值
t = '2323' // Error: 不能将类型""2323""分配给类型""yes, it's a number""。ts(

二、魔法关键字 infer

infer 允许我们在 extends 子句中声明一个待推断的类型变量。如果类型匹配成功,这个变量就会"捕获"对应位置的类型,然后我们可以在条件为真的分支中使用它。

typescript 复制代码
type ReturnType<T> = T extends (...args: any[]) => infer P ? P : T;
interface User {
    name: string;
    age: number;
}
type Func = (name:string) => User;
type pType = ReturnType<Func>; // pType = User
type tType = ReturnType<string>; // tType = string

解析:

  • T 限定判断为一个函数(...args: any[])=> infer P,并且函数的返回类型为P
  • 限定为真时,ReturnType为P
  • 限定为假时,ReturnType为T

三、分布式条件类型

当一个条件类型作用于一个 的泛型参数,且该参数的实际类型是联合类型时,它会被分发到联合类型的每一个成员上进行独立计算,最后将结果联合起来。

typescript 复制代码
type ToArray<T> = T extends string | number ? T[] : T;

type T1 = ToArray<string | number>; // 结果是 string[] | number[]
type T3 = ToArray<boolean>; // 结果是 boolean

type ToArray2<T> = T extends any ? T[] : never;
type T4 = ToArray2<string | number>; // 结果是 string[] | number[]

解析:

  • string | number或者any都能包含 T1T4的联合类型
  • T1T4符合限定,所以返回T[]类型,结果就是分别对联合类型的各个类型进行了数组化处理

例如,我们自己实现一个 Exclude

typescript 复制代码
type MyExclude<T, U> = T extends U ? never : T;

// 'a' | 'b' | 'c' 排除 'a' | 'd'
type Result = MyExclude<'a' | 'b' | 'c', 'a' | 'd'>; // 结果:'b' | 'c'

解析:

  • ('a' extends 'a' | 'd' ? never : 'a') => never
  • ('b' extends 'a' | 'd' ? never : 'b') => 'b'
  • ('c' extends 'a' | 'd' ? never : 'c') => 'c'
  • 最终联合结果:never | 'b' | 'c',即 'b' | 'c'

如何阻止分发? 如果你不希望发生分发,可以用方括号 [] 将泛型参数包裹起来,[T] 就不再是 的泛型参数,不是后就不会分发,而是作为一个整理进行处理:

typescript 复制代码
type ToArrayNonDist<T> = [T] extends [any] ? T[] : never;

type T2 = ToArrayNonDist<string | number>; // 结果是 (string | number)[]

总结

如果你喜欢本教程,记得点赞+收藏!关注我获取更多JavaScript/TypeScript开发干货

相关推荐
一袋米扛几楼986 分钟前
【网络安全】SIEM -Security Information and Event Management 工具是什么?
前端·安全·web安全
小陈工17 分钟前
2026年4月7日技术资讯洞察:下一代数据库融合、AI基础设施竞赛与异步编程实战
开发语言·前端·数据库·人工智能·python
Cobyte26 分钟前
3.响应式系统基础:从发布订阅模式的角度理解 Vue2 的数据响应式原理
前端·javascript·vue.js
竹林81830 分钟前
从零到一:在React前端中集成The Graph查询Uniswap V3池数据实战
前端·javascript
Mintopia37 分钟前
别再迷信"优化":大多数性能问题根本不在代码里
前端
倾颜38 分钟前
接入 MCP,不一定要先平台化:一次 AI Runtime 的实战取舍
前端·后端·mcp
军军君0140 分钟前
Three.js基础功能学习十八:智能黑板实现实例五
前端·javascript·vue.js·3d·typescript·前端框架·threejs
恋猫de小郭40 分钟前
Android 上为什么主题字体对 Flutter 不生效,对 Compose 生效?Flutter 中文字体问题修复
android·前端·flutter
Moment42 分钟前
AI全栈入门指南:一文搞清楚NestJs 中的 Controller 和路由
前端·javascript·后端
禅思院42 分钟前
前端架构演进:基于AST的常量模块自动化迁移实践
前端·vue.js·前端框架