TypeScript 中的 infer 关键字用途
infer 是 TypeScript 中的高级类型关键字,主要用于条件类型中推断类型。它允许我们在条件类型的 extends 子句中声明一个类型变量,然后在该条件类型的 true 分支中使用这个推断出的类型。
1. 基本语法
type SomeType<T> = T extends infer U ? U : never;
2. 主要用途
1. 从复杂类型中提取子类型;
2. 实现类型解构;
3. 创建更灵活的工具类型;
3. 常见示例
3.1. 提取函数返回类型
type ReturnType<T> = T extends (...args: any[]) => infer R ? R : never;
function foo(): number {
return 42;
}
type FooReturn = ReturnType<typeof foo>; // number
3.2. 提取数组/元组元素类型
type ElementType<T> = T extends (infer U)[] ? U : never;
type StrArray = string[];
type StrElement = ElementType<StrArray>; // string
3.3. 提取数组/元组元素类型
type UnpackPromise<T> = T extends Promise<infer U> ? U : T;
type PromiseNumber = Promise<number>;
type NumberType = UnpackPromise<PromiseNumber>; // number
3.4. 提取函数参数类型
type Parameters<T> = T extends (...args: infer P) => any ? P : never;
function bar(a: string, b: number): void {}
type BarParams = Parameters<typeof bar>; // [string, number]
3.5. 提取构造函数实例类型
type InstanceType<T> = T extends new (...args: any[]) => infer R ? R : any;
class MyClass {}
type MyInstance = InstanceType<typeof MyClass>; // MyClass
3.6. 实现类型反转(将元组类型顺序反转)
type Reverse<T> = T extends [infer First, ...infer Rest]
? [...Reverse<Rest>, First]
: T;
type Original = [1, 2, 3];
type Reversed = Reverse<Original>; // [3, 2, 1]
4. 注意事项
1. infer 只能在条件类型的 extends 子句中使用;
2. 同一类型变量可以在多个位置进行 infer,但必须推断出相同的类型;
3. 如果无法推断出类型,则条件类型会返回 never 或指定的默认类型;
infer 是 TypeScript 类型编程中非常强大的工具,它使得类型操作更加灵活和强大,特别是在创建复杂工具类型时非常有用。