AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题 / 77622874
Accepted
jansepke
jansepke
Asked: 2023-12-08 04:16:46 +0800 CST2023-12-08 04:16:46 +0800 CST 2023-12-08 04:16:46 +0800 CST

如何将扩展运算符与打字稿实用程序类型“参数”一起使用

  • 772

我构建了一个高阶函数,如果使用相同的参数调用函数,它会缓存函数的结果。这是使用参数实用程序类型返回具有相同签名的函数,将参数传递给包装函数。

function memoize<FN extends (...args: any) => Promise<any>>(fn: FN) {
  const cache: Record<string, Promise<Awaited<ReturnType<FN>>>> = {};

  return (...args: Parameters<FN>) => {
    const cacheKey = JSON.stringify(args);

    if (!(cacheKey in cache)) {
      // @ts-ignore
      const promise = fn(...args);
      cache[cacheKey] = promise;
    }

    return cache[cacheKey];
  };
}

如果没有ts-ignore打字稿,将会失败Type 'Parameters<FN>' must have a '[Symbol.iterator]()' method that returns an iterator. ts(2488)。我该如何正确解决这个问题?

该函数的使用方式如下:

const cachedFunction = memoize(async (id: string) => fetch(`https://jsonplaceholder.typicode.com/todos/${id}`))
const resultFromAPI = cachedFunction("1")
const resultFromCache = cachedFunction("1")
typescript
  • 1 1 个回答
  • 26 Views

1 个回答

  • Voted
  1. Best Answer
    jcalz
    2023-12-08T04:44:04+08:002023-12-08T04:44:04+08:00

    正如microsoft/TypeScript#36874中报告的那样,不被视为可传播的部分Parameters<FN>被认为是 TypeScript 中的错误。这可以通过将约束从更改为来解决:(...args: any) => ⋯(...args: any[]) => ⋯

    function memoize<FN extends (...args: any[]) => Promise<any>>(fn: FN) {
      const cache: Record<string, Promise<Awaited<ReturnType<FN>>>> = {};
    
      return (...args: Parameters<FN>) => {
        const cacheKey = JSON.stringify(args);
    
        if (!(cacheKey in cache)) {
          const promise = fn(...args); // okay
          cache[cacheKey] = promise;
        }
    
        return cache[cacheKey];
      };
    }
    

    不过,这不是这里推荐的方法。


    Awaited、Parameters和实用程序类型作为条件类型ReturnType实现。在 的体内,类型是泛型类型参数,因此和是泛型条件类型。众所周知,TypeScript 无法对通用条件类型进行大量分析。它往往会推迟对它们的评估,因此在内部,类型大多是不透明的。因此,当您调用编译器时,无法确定是否合适。所以它要么一直报告错误,要么不报告错误。memoize()FNParameters<FN>Awaited<ReturnType<FN>>memoize()Parameters<FN>fn(...args)args

    由于FN被限制为(...args: any[]) => ⋯,调用f(...args)最终会扩大FN到它的约束,它接受任何参数,因此无论如何它最终都无法报告错误。所以fn(...args)被接受了,但是显然有些不好的事情也是如此fn(123):

    const promise = fn(123); // okay?!!!
    

    如果您确实决定使用这种方法,那么您应该非常小心。尽可能避免通用条件类型。


    推荐的方法不是泛型条件类型,而是使您的函数泛型,而不是在FN完整函数类型中,而是在其参数列表A和返回类型中R:

    function memoize<A extends any[], R>(fn: (...args: A) => Promise<R>) {
      const cache: Record<string, Promise<R>> = {};
    
      return (...args: A) => {
        const cacheKey = JSON.stringify(args);
    
        if (!(cacheKey in cache)) {
          const promise = fn(...args); // okay
          cache[cacheKey] = promise;
        }
    
        return cache[cacheKey];
      };
    }
    

    这编译得很干净,编译器实际上能够很好地理解你在做什么,如果你编写,fn(123)你会得到预期的错误

    const promise = fn(123); // error!
    //                 ~~~
    // Argument of type '[number]' is not assignable to parameter of type 'A'.    
    

    Playground 代码链接

    • 1

相关问题

  • 为什么我们在条件语句中使用方括号“[]”?

  • Nestjs有关模块的问题

  • 如何获取文件的打字稿元数据?

  • 如何将数组转换为像这样的对象返回 const 类型?

  • “没有重载匹配”我的 Object.assign() 调用;我该如何修复它?

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    使用 <font color="#xxx"> 突出显示 html 中的代码

    • 2 个回答
  • Marko Smith

    为什么在传递 {} 时重载解析更喜欢 std::nullptr_t 而不是类?

    • 1 个回答
  • Marko Smith

    您可以使用花括号初始化列表作为(默认)模板参数吗?

    • 2 个回答
  • Marko Smith

    为什么列表推导式在内部创建一个函数?

    • 1 个回答
  • Marko Smith

    我正在尝试仅使用海龟随机和数学模块来制作吃豆人游戏

    • 1 个回答
  • Marko Smith

    java.lang.NoSuchMethodError: 'void org.openqa.selenium.remote.http.ClientConfig.<init>(java.net.URI, java.time.Duration, java.time.Duratio

    • 3 个回答
  • Marko Smith

    为什么 'char -> int' 是提升,而 'char -> Short' 是转换(但不是提升)?

    • 4 个回答
  • Marko Smith

    为什么库中不调用全局变量的构造函数?

    • 1 个回答
  • Marko Smith

    std::common_reference_with 在元组上的行为不一致。哪个是对的?

    • 1 个回答
  • Marko Smith

    C++17 中 std::byte 只能按位运算?

    • 1 个回答
  • Martin Hope
    fbrereto 为什么在传递 {} 时重载解析更喜欢 std::nullptr_t 而不是类? 2023-12-21 00:31:04 +0800 CST
  • Martin Hope
    比尔盖子 您可以使用花括号初始化列表作为(默认)模板参数吗? 2023-12-17 10:02:06 +0800 CST
  • Martin Hope
    Amir reza Riahi 为什么列表推导式在内部创建一个函数? 2023-11-16 20:53:19 +0800 CST
  • Martin Hope
    Michael A fmt 格式 %H:%M:%S 不带小数 2023-11-11 01:13:05 +0800 CST
  • Martin Hope
    God I Hate Python C++20 的 std::views::filter 未正确过滤视图 2023-08-27 18:40:35 +0800 CST
  • Martin Hope
    LiDa Cute 为什么 'char -> int' 是提升,而 'char -> Short' 是转换(但不是提升)? 2023-08-24 20:46:59 +0800 CST
  • Martin Hope
    jabaa 为什么库中不调用全局变量的构造函数? 2023-08-18 07:15:20 +0800 CST
  • Martin Hope
    Panagiotis Syskakis std::common_reference_with 在元组上的行为不一致。哪个是对的? 2023-08-17 21:24:06 +0800 CST
  • Martin Hope
    Alex Guteniev 为什么编译器在这里错过矢量化? 2023-08-17 18:58:07 +0800 CST
  • Martin Hope
    wimalopaan C++17 中 std::byte 只能按位运算? 2023-08-17 17:13:58 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve