输入元组,以便功能链具有有效的参数和返回类型

时间:2019-11-18 08:07:11

标签: arrays typescript tuples

我想在下面为此元组/数组创建类型。

这是有效的:

const funcs = [(a: string) => 1, (a: number) => 'A', (a: string) => 2]

这是无效的:

const funcs = [(a: string) => 1, (a: number) => 2, (a: string) => 3]

区别在于中间函数的返回类型从字符串更改为数字。

有可能吗?

type SubtractOne<T extends number> = [-1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62][T];
type AddOne<T extends number> = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62][T];

const funcs = [(a: string) => 1, (a: number) => 'A', (a: string) => 2]

type CheckFuncs<T extends any[]> = { [(K extends number) in keyof T]: T[AddOne<K>] }

type funcsType = typeof funcs
type funcsType2 = CheckFuncs<funcsType>

在研究期间,我遇到了一种使用映射进行索引的方法。是否可以使用它向K添加AddOne或减去K?然后我可以访问ReturnType中的T[K]Parameter<T[k+1]>[0]

Playground

1 个答案:

答案 0 :(得分:0)

在给定索引类型K的情况下,索引到元组的下一个或上一个值是我对tuple rest/spread的支持。考虑Tail<T>,它采用一个元组类型T,并返回一个删除了第一个元素的新元组类型:

// Tail<[1,2,3]> is [2,3]
type Tail<T extends readonly any[]> =
    ((...a: T) => void) extends ((h: any, ...r: infer R) => void) ? R : never;

然后,“ T[K+1]”可以表示为Tail<T>[K]

这可能是您所问问题的答案,但我将继续说明如何使用它:


这是我开始写您的CheckFuncs类型的方法:

type CheckFuncs<T extends readonly ((x: any) => any)[]> = { [K in keyof T]:
    K extends keyof Tail<T> ? (
        [T[K], Tail<T>[K]] extends [(x: infer A) => infer R, (x: infer S) => any] ? (
            [R] extends [S] ? T[K] : (x: A) => S
        ) : never
    ) : T[K]
}

这将遍历T,并将T[K]Tail<T>[K]进行比较,然后将T[K]转换为可以正常工作的版本。然后这些(注意,要保留元组类型,您需要const assertions):

const pass = [(a: string) => 1, (a: number) => 'A', (a: string) => 2] as const
const fail = [(a: string) => 1, (a: number) => 3, (a: string) => 2] as const

产生这些:

type passType = CheckFuncs<typeof pass>
// readonly [(a: string) => number, (a: number) => string, (a: string) => number]

type failType = CheckFuncs<typeof fail>
// readonly [(a: string) => number, (x: number) => string, (a: string) => number]

您可以创建一个与CheckFuncs一起使用的函数,如下所示:

function useFuncs<T extends readonly ((x: any) => any)[]>(...t: CheckFuncs<T>) { }

useFuncs((a: string) => 1, (a: number) => 'A', (a: string) => 2); // okay
useFuncs((a: string) => 1, (a: number) => 3, (a: string) => 2); // error!
// ----------------------> ~~~~~~~~~~~~~~~~
// Argument of type '(a: number) => number' is not assignable to 
// parameter of type '(x: number) => string'.

当然,您想要编写或使用CheckFuncs的确切方法可能有所不同;这只是一个例子。希望能有所帮助;祝你好运!

Link to code