gpt4 book ai didi

带有通用可选参数的 Typescript 函数重载

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

我正在尝试编写一个高阶函数来包装输入函数并将最近调用的结果作为副作用缓存。基本函数( withCache )看起来像这样:

function cache(key: string, value: any) {
//Some caching logic goes here
}

function withCache<R>(key: string, fn: (...args: any[]) => R): (...args: any[]) => R {
return (...args) => {
const res = fn(...args);
cache(key, res);
return res;
}
}

const foo = (x: number, y: number) => x + y;
const fooWithCache = withCache("foo", foo);
let fooResult1 = fooWithCache(1, 2); // allowed :)
let fooResult2 = fooWithCache(1, 2, 3, 4, 5, 6) // also allowed :(

现在我知道我可以使这种类型安全 - 在某种程度上 - 使用函数重载,如下所示:
function withCache<R>(key: string, fn: () => R): () => R
function withCache<R, T1>(key: string, fn: (a: T1) => R): (a: T1) => R
function withCache<R, T1, T2>(key: string, fn: (a: T1, b: T2) => R): (a: T1, b: T2) => R
function withCache<R>(key: string, fn: (...args: any[]) => R): (...args: any[]) => R {
// implementation ...
}

const foo = (x: number, y: number) => x + y;
const fooWithCache = withCache("foo", foo);
let fooResult1 = fooWithCache(1, 2); // allowed :)
let fooResult2 = fooWithCache(1, 2, 3, 4, 5, 6) // not allowed :)

当我尝试允许带有可选参数的函数时出现了问题(最后一个重载是新的):
function withCache<R>(key: string, fn: () => R): () => R
function withCache<R, T1>(key: string, fn: (a: T1) => R): (a: T1) => R
function withCache<R, T1, T2>(key: string, fn: (a: T1, b: T2) => R): (a: T1, b: T2) => R
function withCache<R, T1, T2>(key: string, fn: (a: T1, b?: T2) => R): (a: T1, b?: T2) => R
function withCache<R>(key: string, fn: (...args: any[]) => R): (...args: any[]) => R {
// implementation ...
}

const foo = (x: number, y?: number) => x + (y || 0);
const fooWithCache = withCache("foo", foo);
let fooResult1 = fooWithCache(1); // allowed :)
let fooResult2 = fooWithCache(1, 2) // not allowed, but should be :(

问题似乎是 Typescript 为 withCache 选择了错误的重载,结果是 fooWithCache 的签名是 (a: number) => number .我希望 fooWithCache签名为 (a: number, b?: number) => number ,就像 foo .
有没有什么办法解决这一问题?

(作为旁注,有什么方法可以声明重载,这样我就不必重复每个重载的函数类型 (...) => R 吗?)

编辑:

找出我关于不重复函数类型的第二个问题:只需定义它!
type Function1<T1, R> = (a: T1) => R;
// ...
function withCache<T1, R>(fn: Function1<T1, R>): Function1<T1, R>;

编辑:

这对于异步函数如何工作(假设您想缓存结果而不是 Promise 本身)?你当然可以这样做:
function withCache<F extends Function>(fn: F) {
return (key: string) =>
((...args) =>
//Wrap in a Promise so we can handle sync or async
Promise.resolve(fn(...args)).then(res => { cache(key, res); return res; })
) as any as F; //Really want F or (...args) => Promise<returntypeof F>
}

但是,与同步函数一起使用是不安全的:
//Async function
const bar = (x: number) => Promise.resolve({ x });
let barRes = withCache(bar)("bar")(1).x; //Not allowed :)

//Sync function
const foo = (x: number) => ({ x });
let fooRes = withCache(foo)("bar")(1).x; //Allowed, because TS thinks fooRes is an object :(

有没有办法防范这种情况?或者编写一个对两者都安全有效的函数?

总结:@jcalz 的回答是正确的。在可以假设同步函数的情况下,或者可以直接使用 Promise 而不是它们解析的值的情况下,断言函数类型可能是安全的。但是,如果没有一些目前 unimplemented,上述同步或异步场景是不可能的。 language improvements .

最佳答案

通过向下列表并选择第一个匹配的来选择重载。

检查以下代码,它成功编译:

declare let f: (a: any, b?: any) => void;
declare let g: (a: any) => void;
g = f; // okay

函数 f是一个接受一个或两个参数的函数,而 g被声明为一个接受一个的函数。您可以分配值 f到变量 g ,因为您可以在任何可以调用一个参数的函数的地方调用任何一个或两个参数的函数。正是第二个参数是可选的这一事实使这个赋值起作用。

您还可以执行其他任务:
f = g; //okay

因为你可以在任何可以调用一个或两个参数的函数的地方调用一个参数的任何函数。这意味着这两种类型的函数是可以相互赋值的(即使它们不等价,这有点不健全)。

如果我们只看这两个重载:
function withCache<R, T1>(key: string, fn: (a: T1) => R): (a: T1) => R
function withCache<R, T1, T2>(key: string, fn: (a: T1, b?: T2) => R): (a: T1, b?: T2) => R

以上讨论 fg意味着匹配这些重载之一的任何东西都将匹配另一个。因此,您首先列出的将被选中。你实际上不能同时使用它们,抱歉。

在这一点上,我可以建议您开始提出一组提供合理行为的折衷重载,但让我们备份:

你不只是想要一个类型安全版本的 withCache() ?这个怎么样:
function withCache<F extends Function>(key: string, fn: F): F {     
// implementation ...
}

没有重载,并且返回值总是与 fn 相同的类型范围:
const foo = (x: number, y?: number) => x;
const fooWithCache = withCache("foo", foo); // (x: number, y?: number) => number
let fooResult1 = fooWithCache(1); // allowed :)
let fooResult2 = fooWithCache(1, 2) // allowed :)

那对你有用吗?祝你好运!

关于带有通用可选参数的 Typescript 函数重载,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47208569/

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