簡體   English   中英

通用部分應用程序的 Varargs function 參數不進行類型檢查

[英]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]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的完整集 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

在我看來很好。


好的,希望有幫助; 祝你好運!

Playground 代碼鏈接

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM