1、LengthOfTuple
计算元组类型的长度
type A = LengthOfTuple<['B', 'F', 'E']> // 3type B = LengthOfTuple<[]> // 0
实现:
type LengthOfTuple<T extends any[]> = T extends { length: infer L } ? L : never
2、FirstItem
得到元组类型中的第一个元素
type A = FirstItem<[string, number, boolean]> // stringtype B = FirstItem<['B', 'F', 'E']> // 'B'type C = FirstItem<[]> // never
实现:
type FirstItem<T extends any[]> = T extends [infer First, ...infer Rest] ? First : never
3、LastItem
得到元组类型中的最后一个元素
type A = LastItem<[string, number, boolean]> // booleantype B = LastItem<['B', 'F', 'E']> // 'E'type C = LastItem<[]> // never
实现:
type LastItem<T extends any[]> = T extends [...infer Rest, infer Last] ? Last : never
4、Shift
移除元组类型中的第一个类型
type A = Shift<[1, 2, 3]> // [2,3]type B = Shift<[1]> // []type C = Shift<[]> // []
实现:
type Shift<T extends any[]> = T extends [infer First, ...infer Rest] ? Rest : T
5、Push
在元组类型
T中添加新的类型I
type A = Push<[1,2,3], 4> // [1,2,3,4]type B = Push<[1], 2> // [1, 2]
实现:
type Push<T extends any[], I> = [...T, I]
6、ReverseTuple
反转元组
type A = ReverseTuple<[string, number, boolean]> // [boolean, number, string]type B = ReverseTuple<[1, 2, 3]> // [3,2,1]type C = ReverseTuple<[]> // []
实现:
type ReverseTuple<T extends any[], Prev extends any[] = []> = T extends [...infer Rest, infer Last] ? ReverseTuple<Rest, [...Prev, Last]> : Prev
7、Flat
拍平元组
type A = Flat<[1, 2, 3]> // [1,2,3]type B = Flat<[1, [2, 3], [4, [5, [6]]]]> // [1,2,3,4,5,6]type C = Flat<[]> // []type D = Flat<[1]> // [1]
实现:
type Flat<T extends any[], Prev extends any[] = []> = T extends [infer First, ...infer Rest] ? (First extends any[] ? Flat<Rest, Flat<First, Prev>> : Flat<Rest, [...Prev, First]>) : Prev
8、Repeat
复制类型T为C个元素的元组类型
type A = Repeat<number, 3> // [number, number, number]type B = Repeat<string, 2> // [string, string]type C = Repeat<1, 1> // [1]type D = Repeat<0, 0> // []
实现:
type Repeat<T, C extends number, Prev extends any[] = []> = Prev['length'] extends C ? Prev : Repeat<T, C, [...Prev, T]>
9、Filter
保留元组类型
T中的A类型
type TypeEquals<T, S> = [T] extends [S] ? true : falsetype Filter<T extends any[], A, Prev extends any[] = []> = T extends [infer First, ...infer Rest] ? (TypeEquals<First, A> extends true ? Filter<Rest, A, [...Prev, First]> : Filter<Rest, A, Prev>) : Prev
TypeScript中的Equal的玩法:https://github.com/microsoft/TypeScript/issues/27024
10、FindIndex
找出
E类型在元组类型T中的下标
type A = [any, never, 1, '2', true]type B = FindIndex<A, 1> // 2type C = FindIndex<A, 3> // never
实现:
type Equals<X, Y> =(<T>() => T extends X ? 1 : 2) extends(<U>() => U extends Y ? 1 : 2) ? true : falsetype FindIndex<T extends any[], E, Prev extends any[] = []> = T extends [infer First, ...infer Rest] ? (Equals<First, E> extends true ? Prev['length'] : FindIndex<Rest, E, [...Prev, First]>) : never
小结:
在9、10题中,选择了不同的Equals判断类型,是由类型判断的严格程度决定的。
11、TupleToEnum
元组类型转换为枚举类型
// 默认情况下,枚举对象中的值就是元素中某个类型的字面量类型type a1 = TupleToEnum<["MacOS", "Windows", "Linux"]>// -> { readonly MacOS: "MacOS", readonly Windows: "Windows", readonly Linux: "Linux" }// 如果传递了第二个参数为true,则枚举对象中值的类型就是元素类型中某个元素在元组中的index索引,也就是数字字面量类型type a2 = TupleToEnum<["MacOS", "Windows", "Linux"], true>// -> { readonly MacOS: 0, readonly Windows: 1, readonly Linux: 2 }
实现:
type Equals<X, Y> =(<T>() => T extends X ? 1 : 2) extends(<U>() => U extends Y ? 1 : 2) ? true : falsetype FindIndex<T extends any[], E, Prev extends any[] = []> = T extends [infer First, ...infer Rest] ? (Equals<First, E> extends true ? Prev['length'] : FindIndex<Rest, E, [...Prev, First]>) : nevertype TupleToEnum<T extends string[], V extends boolean = false> = {readonly [K in T[number]]: V extends true ? FindIndex<T, K> : K}
12、Slice
截取元组中的部分元素
type A1 = Slice<[any, never, 1, '2', true, boolean], 0, 2> // [any,never,1] 从第0个位置开始,保留到第2个位置的元素类型type A2 = Slice<[any, never, 1, '2', true, boolean], 1, 3> // [never,1,'2'] 从第1个位置开始,保留到第3个位置的元素类型type A3 = Slice<[any, never, 1, '2', true, boolean], 1, 2> // [never,1] 从第1个位置开始,保留到第2个位置的元素类型type A4 = Slice<[any, never, 1, '2', true, boolean], 2> // [1,'2',true,boolean] 从第2个位置开始,保留后面所有元素类型type A5 = Slice<[any], 2> // [] 从第2个位置开始,保留后面所有元素类型type A6 = Slice<[], 0> // [] 从第0个位置开始,保留后面所有元素类型
实现:
type TupleSplit<T extends any[], N extends number, O extends any[] = []> =O['length'] extends N ? [O, T] : T extends [infer First, ...infer Rest] ?TupleSplit<Rest, N, [...O, First]> : [O, T]type TakeFirst<T extends any[], N extends number> = TupleSplit<T, N>[0]type SkipFirst<T extends any[], N extends number> = TupleSplit<T, N>[1]type Slice<T extends any[], Start extends number, End extends number = 0> =End extends 0 ? SkipFirst<TakeFirst<T, T['length']>, Start> :SkipFirst<[...TakeFirst<T, End>, ...TakeFirst<SkipFirst<T, End>, 1>], Start>
13、Splice
删除并且替换部分元素
type A1 = Splice<[string, number, boolean, null, undefined, never], 0, 2> // [boolean,null,undefined,never] 从第0开始删除,删除2个元素type A2 = Splice<[string, number, boolean, null, undefined, never], 1, 3> // [string,undefined,never] 从第1开始删除,删除3个元素type A3 = Splice<[string, number, boolean, null, undefined, never], 1, 2, [1, 2, 3]> // [string,1,2,3,null,undefined,never] 从第1开始删除,删除2个元素,替换为另外三个元素1,2,3
实现:
type TupleSplit<T extends any[], N extends number, O extends any[] = []> =O['length'] extends N ? [O, T] : T extends [infer First, ...infer Rest] ?TupleSplit<Rest, N, [...O, First]> : [O, T]type TakeFirst<T extends any[], N extends number> = TupleSplit<T, N>[0]type SkipFirst<T extends any[], N extends number> = TupleSplit<T, N>[1]type Splice<T extends any[], S extends number, E extends number, A extends any[] = []> =A extends [] ? [...TakeFirst<T, S>, ...SkipFirst<SkipFirst<T, S>, E>] : [...TakeFirst<T, S>, ...A, ...SkipFirst<SkipFirst<T, S>, E>]
