模板文字类型打字稿重复

Kou*_*sha 9 typescript

是否可以使用模板文字类型构建重复?例如:

type Hex = 'a' | 'b' | 'c'| ...;
type Id = `${Hex}${Hex}...` // Here I want to say Id is N hex long. 
Run Code Online (Sandbox Code Playgroud)

A_b*_*lop 9

原则上,这可以通过 TS 4.1 中的递归条件类型实现:

type Decrement = [never, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
type RepeatString<S extends string, N extends number> =
    N extends 1 ? S :
    `${S}${RepeatString<S, Decrement[N]>}`
Run Code Online (Sandbox Code Playgroud)
type T11 = Decrement[5] // 4
type T12 = RepeatString<"foo", 3> // "foofoofoo"
type T13 = RepeatString<Hex, 3> // "aaa" | "aab" | ... | "ccb" | "ccc"
Run Code Online (Sandbox Code Playgroud)

请注意以下限制

请注意,联合类型的跨产品分布可能会迅速升级为非常大且成本高昂的类型。另请注意,工会类型仅限于少于 100,000 个成员[.](我的重点)

例如,使用Hex2being 'a' | 'b' | ... | 'i' | 'j'(10个联合部分),type T14 = RepeatString<Hex2, 5>将触发以下错误(10*10*10*10*10排列):

表达式生成的联合类型过于复杂而无法表示。(2590)

在这种情况下,需要减少联合部分的数量,例如寻找更大的字符串块而不是原子字符。

选择

type RepeatStringAlt<S extends string, N extends number> = RepeatStringAltRec<S, TupleOf<unknown, N>>
type RepeatStringAltRec<S extends string, T extends unknown[]> =
    T["length"] extends 1 ? S : `${S}${RepeatStringAltRec<S, DropFirst<T>>}`
    
type TupleOf<T, N extends number> = N extends N ? number extends N ? T[] : _TupleOf<T, N, []> : never;
type _TupleOf<T, N extends number, R extends unknown[]> = R['length'] extends N ? R : _TupleOf<T, N, [T, ...R]>;
type DropFirst<T extends readonly unknown[]> = T extends readonly [any?, ...infer U] ? U : [...T];

type T23 = RepeatStringAlt<"foo", 3> // "foofoofoo"
Run Code Online (Sandbox Code Playgroud)

这摆脱了手动计算数字范围的麻烦Decrement

实时代码游乐场