[英]Varargs function argument of generic partial application does not type check
我有以下组合器,它将多参数 function 转换为可以部分应用的组合器:
type Tuple = any[];
const partial = <A extends Tuple, B extends Tuple, C>
(f: (...args: (A & B)[]) => C, ...args1: A) => (...args2: B) =>
// ^^^^^^^^^^^^^^^^^^
f(...args1, ...args2);
const sum = (v: number, w: number, x: number, y: number, z: number) =>
w + w + x + y + z;
partial(sum, 1, 2, 3)(4, 5);
// ^^^
这不起作用,因为 function 参数f
必须接受不同数量的 arguments 而不使用 rest 语法。 有没有办法输入f
?
您不能通过相交来连接元组类型。 例如, [string, number] & [boolean]
不等同于[string, number, boolean]
。 相反,它是一个不可能的元组,其length
是不可居住类型1 & 2
,其第一个元素是不可居住类型string & boolean
。 在类型级别没有内置的元组连接(请参阅microsoft/TypeScript#5453 ),并且变通办法有各种丑陋和不受支持的风格。
这是一个有点丑陋且可能不受支持的解决方法(尽管请参阅microsoft/TypeScript#32131 ,它将为Array.flat()
引入几乎相同的新类型):
type Prev = [never, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11];
type Tail<T extends any[]> =
((...t: T) => void) extends ((h: any, ...t: infer R) => void) ? R : never;
type Drop<T extends any[], N extends number> =
{ 0: T, 1: Drop<Tail<T>, Prev[N]> }[N extends 0 ? 0 : 1];
const partial = <
X extends any[],
Y extends Extract<{ [K in keyof Y]: K extends keyof X ? X[K] : never }, any[]>,
R>(
f: (...args: X) => R,
...args1: Y
) => (...args2: Drop<X, Y['length']>): R => f(...[...args1, ...args2] as any);
Prev
类型只是一个元组,它可以让你从一个数字到前一个数字,直到你想要的任何限制。 所以Prev[4]
是3
而Prev[3]
是2
。
Tail<T>
类型采用元组类型T
并剥离第一个元素,留下后面的所有元素。 所以Tail<[1, 2, 3, 4]>
是[2, 3, 4]
。
Drop<T, N>
类型是可能不受支持的递归事物,它采用元组类型T
和数字N
并剥离前N
个元素,将所有内容留在后面。 所以Drop<T, 1>
基本上就是Tail<T>
,而Drop<[1, 2, 3, 4, 5], 2>
是[3, 4, 5]
。
最后, partial()
签名在元组类型X
中是通用的,对应于f
的完整集 arguments 和一个元组类型Y
,对应于 ZDBC11CAA5BDA99F77E6FB4DABD8 的 rest 的Y
,并且必须是部分段的 Y82E partial()
X
。 所以如果x
是[1,2,3,4,5]
,那么Y
可以是[1]
,或[1, 2]
,...或[1, 2, 3, 4, 5]
。 R
类型是f
的返回类型。 然后,它返回一个新的 function ,其返回类型为R
,其参数类型为Drop<X, Y['length']>
。 也就是说,返回的 function 在Y
中的之后接受 arguments 到f
。
让我们看看它是否有效:
const sum = (v: number, w: number, x: number, y: number, z: number) => v + w + x + y + z;
const okay = partial(sum, 1, 2, 3); // const okay: (y: number, z: number) => number
console.log(okay(4, 5)) // 15
const bad = partial(sum, "a", "b", "c"); // error "a" is not number
const alsoBad = partial(sum, 1, 2, 3, 4, 5, 6); // error 6 is not never
在我看来很好。
好的,希望有帮助; 祝你好运!
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.