gpt4 book ai didi

javascript - 通用部分应用程序的 Varargs 函数参数不进行类型检查

转载 作者:行者123 更新时间:2023-12-02 20:53:46 24 4
gpt4 key购买 nike

我有以下组合器,它将多参数函数转换为可以部分应用的函数:

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);
// ^^^

Playground

这不起作用,因为函数参数 f 必须接受不同数量的参数而不使用剩余语法。有没有办法输入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]3Prev[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 的完整参数集,以及元组类型 Y ,对应于 partial() 的其余参数,和Y必须是 X 的某个初始段。所以如果 x[1,2,3,4,5] ,然后Y可以是[1] ,或[1, 2] , ... 或 [1, 2, 3, 4, 5] 。类型Rf 的返回类型。然后,它返回一个新函数,其返回类型为 R ,其参数类型为 Drop<X, Y['length']> 。也就是说,返回的函数接受 f 的参数。 Y 中的内容之后.

<小时/>

让我们看看它是否有效:

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

我觉得不错。

<小时/>

好的,希望有帮助;祝你好运!

Playground link to code

关于javascript - 通用部分应用程序的 Varargs 函数参数不进行类型检查,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61539318/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com