gpt4 book ai didi

typescript :重载函数的返回类型

转载 作者:行者123 更新时间:2023-12-04 13:57:35 25 4
gpt4 key购买 nike

ReturnType 给出的类型似乎取决于写入重载签名的顺序

function applyChanges1(input: string): number
function applyChanges1(input: number): string
function applyChanges1(input: number | string): number | string {
return typeof input === "number" ? input.toString() : input.length
}

function applyChanges2(input: number): string
function applyChanges2(input: string): number
function applyChanges2(input: number | string): number | string {
return typeof input === "number" ? input.toString() : input.length
}

type Ret1 = ReturnType<typeof applyChanges1> // string
type Ret2 = ReturnType<typeof applyChanges2> // number

它似乎采用了最后一个重载签名的返回类型,这似乎很随意。我期待两个 Ret1Ret2成为 string | number .这种行为有原因吗?

最佳答案

正如 Matt McCutchen 所指出的,这是 ReturnType 的限制。以及一般条件类型和多个重载签名。

然而,我们可以构造一个类型,该类型将返回最多任意数量的重载的所有重载返回类型:

function applyChanges1(input: string): number
function applyChanges1(input: number): string
function applyChanges1(input: number | string): number | string {
return typeof input === "number" ? input.toString() : input.length
}

function applyChanges2(input: number): string
function applyChanges2(input: string): number
function applyChanges2(input: number | string): number | string {
return typeof input === "number" ? input.toString() : input.length
}


type OverloadedReturnType<T> =
T extends { (...args: any[]) : infer R; (...args: any[]) : infer R; (...args: any[]) : infer R ; (...args: any[]) : infer R } ? R :
T extends { (...args: any[]) : infer R; (...args: any[]) : infer R; (...args: any[]) : infer R } ? R :
T extends { (...args: any[]) : infer R; (...args: any[]) : infer R } ? R :
T extends (...args: any[]) => infer R ? R : any


type RetO1 = OverloadedReturnType<typeof applyChanges1> // string | number
type RetO2 = OverloadedReturnType<typeof applyChanges2> // number | string

上面的版本最多可用于 4 个重载签名(无论它们是什么),但可以轻松(如果不是很漂亮)扩展到更多。

我们甚至可以用同样的方式得到可能参数类型的联合:
type OverloadedArguments<T> = 
T extends { (...args: infer A1) : any; (...args: infer A2) : any; (...args: infer A3) : any ; (...args: infer A4) : any } ? A1|A2|A3|A4 :
T extends { (...args: infer A1) : any; (...args: infer A2) : any; (...args: infer A3) : any } ? A1|A2|A3 :
T extends { (...args: infer A1) : any; (...args: infer A2) : any } ? A1|A2 :
T extends (...args: infer A) => any ? A : any


type RetO1 = OverloadedArguments<typeof applyChanges1> // [string] & [number]
type RetO2 = OverloadedArguments<typeof applyChanges2> // [number] & [string]

关于 typescript :重载函数的返回类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58773217/

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