likes
comments
collection
share

十三套 TypeScript 类型体操练习

作者站长头像
站长
· 阅读数 63

前面几篇文章介绍了 typescript 类型体操的基本动作,这次我会将基本体操与 Ts 中的工具类型结合起来,并一一实现它。通过实践来深刻体会下如何将几个基础体操动作结合在一起,完成一套漂亮的动作。

程序和体操都需要经过实践、反馈、纠正才能真正掌握。

Just Do It!

十三套 TypeScript 类型体操练习

Exclude<UniType, ExcludeMembers>

功能:

可以通过 UnionType 去除与 ExcludedMembers 中匹配的联合类型来构造类型。

代码示例:

type T0 = Exclude<"a" | "b" | "c", "a">; // T0 = "b" | "c"
     
type T1 = Exclude<"a" | "b" | "c", "a" | "b">; // T1 = "c"
     
type T2 = Exclude<string | number | (() => void), Function>; // T2 = string | number

type Shape =
  | { kind: "circle"; radius: number }
  | { kind: "square"; x: number }
  | { kind: "triangle"; x: number; y: number };
 
type T3 = Exclude<Shape, { kind: "circle" }>

// 相当于
type T3 = {
    kind: "square";
    x: number;
} | {
    kind: "triangle";
    x: number;
    y: number;
}

基本体操动作:

  • 联合类型与交叉类型的区别

  • 泛型,通过泛型参数接受要处理的类型

  • extends关键字结合条件判断

  • never 关键字

实现 MyExclude,前置基础与思路分析:

  1. 两个不想交的基本类型进行相交操作为 never

    type Str = "a" & "c" // Str = never
    
  2. 两个相同的基本类型进行相交操作为其本身

    type Str = "a" & "a" // Str = "a"
    
  3. 两个相同的基本类型进行联合操作为其本身

    type Str = "a" | "a" // Str = "a"
    
  4. never 为其他类型的子类型,任何类型与 never 进行联合操作,返回其本身

    type Str = "a" | never // Str = "a"
    
  5. 需要知道**条件类型在处理处理泛型时,如果传递的泛型为一个联合类型,则会触发类型分配**

    type ToArray<Type> = Type extends any ? Type[] : never;
    
    type ToArray<Type> = Type extends any ? Type[] : never;
     
    type StrArrOrNumArr = ToArray<string | number>; // StrArrOrNumArr = string[] | number[]
    

    相当于在泛型为联合类型时,条件类型对联合类型中的每个类型做了遍历操作。

  6. MyExclude 接受两个泛型参数,使用 extends 关键字进行判断,因为需要保留的是 U 中的类型, E extends U 必为真,所以需要进行 U extends E 的判断。因为

type A = "a" | "b" | "c"
type B = "a"

type MyExclude<U, E> =  U extends E ?never : U

type C = MyExclude<A, B> // C = 'b' | 'c'

type T0 = MyExclude<"a" | "b" | "c", "a">; // T0 = "b" | "c"
     
type T1 = MyExclude<"a" | "b" | "c", "a" | "b">; // T1 = "c"
     
type T2 = MyExclude<string | number | (() => void), Function>; // T2 = string | number

type Shape =
  | { kind: "circle"; radius: number }
  | { kind: "square"; x: number }
  | { kind: "triangle"; x: number; y: number };
 
type T3 = MyExclude<Shape, { kind: "circle" }>

Extract<Type, Union>

功能:

Exclude 相反,保留两个联合类型中的交集部分。

代码示例:

type T0 = Extract<"a" | "b" | "c", "a" | "f">; // type T0 = "a"
type T1 = Extract<string | number | (() => void), Function>; // type T1 = () => void
 
type Shape =
  | { kind: "circle"; radius: number }
  | { kind: "square"; x: number }
  | { kind: "triangle"; x: number; y: number };
 
type T2 = Extract<Shape, { kind: "circle" }>

// 相当于
type T2 = {
    kind: "circle";
    radius: number;
}

基本体操动作:

  • 联合类型与交叉类型的区别

  • 泛型,通过泛型参数接受要处理的类型

  • extends关键字结合条件判断
  • never 关键字

实现 MyExtract

  • 思路与 MyExclude 相似,不同之处在于 extends 关键字进行判断时,是谁在前谁在后的问题。
type MyExtract<T, U> = T extends U ? T : never

type T = MyExtract<"a" | "b" | "c", "a">; // T0 = "b" | "c"
     
type T1 = MyExtract<"a" | "b" | "c", "a" | "b">; // T1 = "c"
     
type T2 = MyExtract<string | number | (() => void), Function>; // T2 = string | number

type Shape =
  | { kind: "circle"; radius: number }
  | { kind: "square"; x: number }
  | { kind: "triangle"; x: number; y: number };
 
type T3 = MyExtract<Shape, { kind: "circle" }>

Partial<Type>

功能:

通过将一个类型对象的所有属性设置为可选状态来构造一个新的类型对象。

代码示例:

interface Todo {
  title: string;
  description: string;
}
 
function updateTodo(todo: Todo, fieldsToUpdate: Partial<Todo>) {
  return { ...todo, ...fieldsToUpdate };
}
 
const todo1 = {
  title: "organize desk",
  description: "clear clutter",
};
 
const todo2 = updateTodo(todo1, {
  description: "throw out trash",
});

基本体操动作

  • ? 属性修饰符,可配置属性是否可选
  • keyof 关键字,可以获取类型对象的所有属性键的联合类型
  • in 关键字,可实现对联合类型的遍历
  • 索引类型,可以获取类型对象的属性值类型

实现 MyPartial,前置基础与思路分析:

  • 通过 keyof 获取泛型 T 的所有属性的联合类型
  • 通过遍历联合类型并添加 ? 操作符,将其转为可选状态
  • 通过 T[Key] 的形式获取属性值类型
type MyPartial<T> {
	[Key in keyof T]?: T[Key] | undefined;
}
function updateTodo(todo: Todo, fieldsToUpdate: MyPartial<Todo>) {
  return { ...todo, ...fieldsToUpdate };
}
 
const todo1 = {
  title: "organize desk",
  description: "clear clutter",
};
 
const todo2 = updateTodo(todo1, {
  description: "throw out trash",
});

Required<Type>

功能:

Partial<Type> 功能相反,通过将一个存在可选属性的类型对象,全部转为必选属性来构造一个新类型对象。

代码示例:

interface Props {
  a?: number;
  b?: string;
}
 
const obj: Props = { a: 5 };
 
const obj2: Required<Props> = { a: 5 }; // Error: Property 'b' is missing in type '{ a: number; }' but required in type 'Required<Props>'.

基本体操动作:

  • 要知道,在 ts 类型对象属性的 ?readonly 前添加 -+ 可实现对类型修饰符的去与留。
  • 通过 keyof 获取泛型 T 的所有属性的联合类型
  • 通过遍历联合类型并添加 ? 操作符,将其转为可选状态
  • 通过 T[Key] 的形式获取属性值类型

实现 MyRequired<Type>

  • 实现思路与 Partial<Type> 相同,不同的是需要通过在属性修饰符前加 -? 移除属性的可选特性。
type MyRequired<T> = {
  [Key in keyof T]-?: T[Key]
}

interface Props {
  a?: number;
  b?: string;
}
 
const obj: Props = { a: 5 };
 
const obj2: MyRequired<Props> = { a: 5 }; // Error: Property 'b' is missing in type '{ a: number; }' but required in type 'Required<Props>'.


Readonly<Type>

功能:

通过将一个类型的所有属性设置为只读(readonly)来构造一个新的类型对象。

代码示例:

interface Todo {
  title: string;
}
 
const todo: Readonly<Todo> = {
  title: "Delete inactive users",
};
 
todo.title = "Hello"; // Cannot assign to 'title' because it is a read-only property.

基本体操动作:

  • Required<Type> 相同

实现 MyReadonly<T>

type MyReadonly<T> = {
  readonly [Key in keyof T]: T[Key]
}
const todo: MyReadonly<Todo> = {
  title: "Delete inactive users",
};
 
todo.title = "Hello"; // Error

Pick<Type, Keys>

功能:

通过从类型对象中选取一组属性 Keys 来构造一个新的类型。

interface Todo {
  title: string;
  description: string;
  completed: boolean;
}
 
type TodoPreview = Pick<Todo, "title" | "completed">;
 
const todo: TodoPreview = {
  title: "Clean room",
  completed: false,
};

基本体操动作

  • 泛型,通过泛型接受类型对象和所需 key 的集合
  • keyof 关键字,传入的 Keys 应该是 T 的所有属性的联合类型的子类型,需要通过 keyof 获取到 T 的所有属性联合类型
  • extends 关键字,通过 extends 实现对 Keys 的约束。
  • 下标类型 & in 关键字,通过 in 关键字实现对 Keys 的遍历构建 key 与属性值类型之间的映射。

实现 MyPick<T, Keys>

type MyPick<T, Keys extends keyof T> = {
  [Key in Keys]: T[Key]
}

type TodoPreview = MyPick<Todo, "title" | "completed">;
 
const todo: TodoPreview = {
  title: "Clean room",
  completed: false,
};

Omit<Type, Keys>

功能:

通过从 Type 中选取所有属性的联合类型,然后移除与 Keys 相交的属性来构造类型。Pick 的反义词。

代码示例:

interface Todo {
  title: string;
  description: string;
  completed: boolean;
  createdAt: number;
}

// 移除 description 属性
type TodoPreview = Omit<Todo, "description">;

const todo: TodoPreview = {
  title: "Clean room",
  completed: false,
  createdAt: 1615544252770,
};

// 移除 "completed" | "createdAt" 属性
type TodoInfo = Omit<Todo, "completed" | "createdAt">;

const todoInfo: TodoInfo = {
  title: "Pick up kids",
  description: "Kindergarten closes at 5pm",
};

基本体操动作:

  • 泛型,通过泛型接受类型参数
  • extends 关键字,通过 extends 关键字实现对 Keys 的约束
  • in 关键字,通过 in 关键字实现对 筛选出来的 符合需求的联合类型遍历
  • keyof 关键字,通过 keyof 获取 T 的属性联合类型
  • 下标类型,通过下标类型实现属性和值得读取操作

实现 MyOmit<T, Keys>

  • 需要基于 MyExclude<T, Keys> 过滤出符合需求的属性联合类型
  • 在基于符合需求的联合类型进行遍历操作,读取属性值类型
type MyOmit<T, Keys extends keyof T> = {
  [Key in MyExclude<keyof T, Keys>]: T[Key]
}

// 移除 description 属性
type TodoPreview = MyOmit<Todo, "description">;

const todo: TodoPreview = {
  title: "Clean room",
  completed: false,
  createdAt: 1615544252770,
};

// 移除 "completed" | "createdAt" 属性
type TodoInfo = MyOmit<Todo, "completed" | "createdAt">;

const todoInfo: TodoInfo = {
  title: "Pick up kids",
  description: "Kindergarten closes at 5pm",
};

补充:

  • 上面的 MyOmit<T, Keys> 基本符合了我的需求,我这里又看了下 Ts 内部是如何实现 Omit<T, Keys> 的:
type Omit<T, Keys extends string | number | symbol> = {
  [Key in MyExclude<keyof T, Keys>]: T[Key]
}

注意到原生的 Omit<T, Keys> 对于 Keys 约束更加宽泛一些,我们自己实现的 MyOmit<T, Keys> 直接将 Keys 约束到 泛型 T 的属性联合类型中。

NonNullable<Type>

功能:

通过从 Type 中移除 nullundefined 来构造一个新的类型对象。

示例代码:

type T0 = NonNullable<string | number | undefined>; // T0 = string | number

type T1 = NonNullable<string[] | null | undefined>; // T1 = string[]

基本体操动作

  • 泛型
  • 交叉类型

实现

type MyNonNullable<T> = T & {}

Parameters<Type>

功能

通过函数类型 Type 的形参类型构造一个元组类型。

代码示例:

declare function f1(arg: { a: number; b: string }): void;
 
type T0 = Parameters<() => string>; // T0 = []
type T1 = Parameters<(s: string) => void>; // T1 = [s: string]
type T2 = Parameters<<T>(arg: T) => T>; // T2 = [arg: unknown]
type T3 = Parameters<typeof f1>;
// 相当于
// type T3 = [arg: {
//    a: number;
//     b: string;
// }]
type T4 = Parameters<any>; // T4 = unknown[]
type T5 = Parameters<never>; // T5 = never
type T6 = Parameters<string>; // Error: Type 'string' does not satisfy the constraint '(...args: any) => any'.  T6 = never

基本体操动作

  • 泛型

  • extends 关键字,通过 extends 对泛型参数 T 进行类型约束,并进行条件类型判断

  • infer 关键字,使用 infer 关键字声明函数的参数类型 P ,并在真条件分支中返回

    • infer 关键字,可以在 extends 条件语句中声明待推断的类型变量。例如:

      type ArrayElementType<T> = T extends (infer E) [] ? E : T; 可以显示对数组元素的推导。

    • 哪些是待推断的类型?其实这取决与约束条件,如果是对象类型,基本都可以转为待推断的,比如函数的参数、函数的返回值、数组的内部元素类型、泛型对象类型的泛型参数。

实现 MyParameters<T>

  • MyParameters<T> 的泛型参数尽可以推断函数类型的参数类型。
  • 需要将 T 约束为函数类型:T extends (...args: any) => any
  • 这里使用 any 的原因是,any 是所有类型的父类型,它可以覆盖 T 的所有情况
  • 使用 extends 关键字对 T 进行约束,并使用 infer 关键字对参数类型进行推导声明
type MyParameters<T extends (...args: any) => any> = T extends (...args: infer P) => any ? P : never

type T0 = MyParameters<() => string>; // T0 = []
type T1 = MyParameters<(s: string) => void>; // T1 = [s: string]
type T2 = MyParameters<<T>(arg: T) => T>; // T2 = [arg: unknown]
type T3 = MyParameters<typeof f1>;
// 相当于
// type T3 = [arg: {
//    a: number;
//     b: string;
// }]
type T4 = MyParameters<any>; // T4 = unknown[]
type T5 = MyParameters<never>; // T5 = never
type T6 = MyParameters<string>; // Error

趁热打铁,接下来实现 ReturnType<Type>

ReturnType<Type>

功能:

通过函数类型 Type 的返回值类型来构造一个新的类型。

代码示例:

declare function f1(): { a: number; b: string };
 
type T0 = ReturnType<() => string>; // type T0 = string
type T1 = ReturnType<(s: string) => void>; // type T1 = void
type T2 = ReturnType<<T>() => T>; // type T2 = unknown
type T3 = ReturnType<<T extends U, U extends number[]>() => T>; // type T3 = number[]
type T4 = ReturnType<typeof f1>;
// 相当于 
type T4 = {
    a: number;
    b: string;
}
type T5 = ReturnType<any>; // type T5 = any
type T6 = ReturnType<never>; // type T6 = never
type T7 = ReturnType<string>; // Error

基本体操动作

  • Parameters<T> 一致

实现 MyReturnType<T>

思路与 Parameters<T> 一样,需要注意 infer 关键字的推导位置

type MyReturnType<T extends (...args: any) => any> = T extends (...arg: any) => infer R ? R : never

type T0 = MyReturnType<() => string>; // type T0 = string
type T1 = MyReturnType<(s: string) => void>; // type T1 = void
type T2 = MyReturnType<<T>() => T>; // type T2 = unknown
type T3 = MyReturnType<<T extends U, U extends number[]>() => T>; // type T3 = number[]
type T4 = MyReturnType<typeof f1>;

type T5 = MyReturnType<any>; // type T5 = any
type T6 = MyReturnType<never>; // type T6 = never
type T7 = MyReturnType<string>; // Error

ConstructorParameters<Type>

功能:

Parameters<T> 类似,不同之处在于可以通过从构造函数类型 Type 的参数类型构造一个元组或者数组类型,如果 Type 不是函数类型,则返回 never 类型。

代码示例:

type T0 = ConstructorParameters<ErrorConstructor>; // type T0 = [message?: string]
type T1 = ConstructorParameters<FunctionConstructor> // type T1 = string[]
type T2 = ConstructorParameters<RegExpConstructor>; // type T2 = [pattern: string | RegExp, flags?: string]
class C {
  constructor(a: number, b: string) {}
}
type T3 = ConstructorParameters<typeof C>; // type T3 = [a: number, b: string]
type T4 = ConstructorParameters<any>; // type T4 = unknown[]

type T5 = ConstructorParameters<Function>; // Error type T5 = never

基本体操动作:

  • Parameters<T> 不同的是需要对 T 的约束类型不是一个函数,而是一个抽象的构造函数。
  • 声明构造函数类型:abstract new (...args: any) => any

实现 MyConstructorParameters<T>

type MyConstructorParameters<T extends abstract new (...args: any) => any> = T extends abstract new (...args: infer P) => any ? P : never

type T0 = MyConstructorParameters<ErrorConstructor>; // type T0 = [message?: string]
type T1 = MyConstructorParameters<FunctionConstructor> // type T1 = string[]
type T2 = MyConstructorParameters<RegExpConstructor>; // type T2 = [pattern: string | RegExp, flags?: string]
class C {
  constructor(a: number, b: string) {}
}
type T3 = MyConstructorParameters<typeof C>; // type T3 = [a: number, b: string]
type T4 = MyConstructorParameters<any>; // type T4 = unknown[]

type T5 = MyConstructorParameters<Function>; // Error type T5 = never

InstanceType<Type>

功能:

构造一个由构造函数 Type 的实例类型组成的类型。相当与 ReturnType<T>

InstanceType 相对于 ConstructorParameters<Type> 的关系,相当与 ReturnType<T> 相对于 Parameters<Type> 的关系。

示例:

class C {
  x = 0;
  y = 0;
}

type T0 = InstanceType<typeof C>; // type T0 = C
type T1 = InstanceType<any>; // type T1 = any
type T2 = InstanceType<never>; // type T2 = never
type T3 = InstanceType<string>; // Type 'string' does not satisfy the constraint 'abstract new (...args: any) => any'. type T3 = any
type T4 = InstanceType<Function>; // type T4 = any

基本体操动作:

  • ConstructorParameters<Type> 相同

实现 MyInstanceType<Type>:

type MyInstanceType<T extends abstract new (...args: any) => any> = T extends abstract new (...args: any) => infer P ? P : never

type T0 = MyInstanceType<typeof C>; // type T0 = C
type T1 = MyInstanceType<any>; // type T1 = any
type T2 = MyInstanceType<never>; // type T2 = never
type T3 = MyInstanceType<string>; // Type 'string' does not satisfy the constraint 'abstract new (...args: any) => any'. type T3 = any
type T4 = MyInstanceType<Function>; // type T4 = any

Awaited<Type>

功能:

该类型用于模拟 async 函数中的 await 操作,或者 promise 上的 .then() 方法,独特之处在于,它们通过递归展开 promise 的方式获取异步返回的结果。

type A = Awaited<Promise<string>>; // type A = string
type B = Awaited<Promise<Promise<number>>>; // type B = number
type C = Awaited<boolean | Promise<number>>; // type C = number | boolean

基本体操动作:

  • 泛型,通过泛型接受异步函数
  • extends 关键字,通过 extends 关键字进行条件类型判断,条件类型中并不支持 if else 语句,需要重复使用 extends 进行判断。
  • infer 关键字,通过 infer 关键字,推断 .then(onfulfilled: infer F) 的参数 onfulfilled 函数类型和该函数返回值的类型。
  • 鸭子辨型,通过鸭子辨型判断是否是异步对象。

实现 MyAwaited<T>:

  • 鸭子辨型判断 T 泛型结构:T extends object & { then(onfulfilled: infer F): any; } ? TrueBranchType : FalseBranchType
  • 如果 推断的类型 F 仍未函数类型,则继续递归 F extends (value: infer V) => any ? Awaited<V> : never
type MyAwaited<T> = T extends null | undefined ? T : T extends object & {
    then(onfulfilled: infer F): any;
} ? F extends (value: infer V) => any ? Awaited<V> : never : T

type A = MyAwaited<Promise<string>>; // type A = string
type B = MyAwaited<Promise<Promise<number>>>; // type B = number
type C = MyAwaited<boolean | Promise<number>>; // type C = number | boolean

总结

十三套体操到此结束。最后希望,大家都能在 playground: www.typescriptlang.org/play 中动手敲一敲,才能加深理解啊。

完~

有收获的老铁,还望点赞、收藏 + 评论啊。

参考:

转载自:https://juejin.cn/post/7238552719266463805
评论
请登录