解锁 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开发干货

相关推荐
大模型玩家七七9 分钟前
梯度累积真的省显存吗?它换走的是什么成本
java·javascript·数据库·人工智能·深度学习
. . . . .17 分钟前
shadcn组件库
前端
2501_9447114325 分钟前
JS 对象遍历全解析
开发语言·前端·javascript
发现一只大呆瓜1 小时前
虚拟列表:支持“向上加载”的历史消息(Vue 3 & React 双版本)
前端·javascript·面试
css趣多多1 小时前
ctx 上下文对象控制新增 / 编辑表单显示隐藏的逻辑
前端
阔皮大师1 小时前
INote轻量文本编辑器
java·javascript·python·c#
lbb 小魔仙1 小时前
【HarmonyOS实战】React Native 表单实战:自定义 useReactHookForm 高性能验证
javascript·react native·react.js
_codemonster1 小时前
Vue的三种使用方式对比
前端·javascript·vue.js
寻找奶酪的mouse1 小时前
30岁技术人对职业和生活的思考
前端·后端·年终总结
梦想很大很大1 小时前
使用 Go + Gin + Fx 构建工程化后端服务模板(gin-app 实践)
前端·后端·go