AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / coding / Perguntas / 79555663
Accepted
Harald
Harald
Asked: 2025-04-04 23:00:08 +0800 CST2025-04-04 23:00:08 +0800 CST 2025-04-04 23:00:08 +0800 CST

Digitação de função genérica para criar n-tuplas de parâmetros e chamar um consumidor

  • 772

Considere o seguinte pseudo-TypeScript, onde Classdenota algo parecido com um construtor:

function convert<C extends Class>(
  stuff: unknown,
  mappers: [Mapper<unknown, parameter i of constructor C>],
  c: constructor of C
): C {
  return c(...mappers.map((f) => f(stuff))
}

A ideia é que cada um mapperextraia um parâmetro do construtor stuffe a tupla de resultado seja passada ao construtor.

Nesta forma simples seria um tanto bobo, pois eu sempre poderia chamar o construtor como new C(map1(stuff), map2(stuff), ...)em vez de chamar esta construção elaborada. Mas uma vez que eu tenha o acima, eu quero adicionar tratamento de erro para os mapeadores de forma que o construtor seja chamado somente se nenhum tiver um problema.

O que tenho até agora:

type Class = abstract new (...args: any) => any;
type Func<A, C> = (a: A) => C;
type FuncTuple<A, TUP> = { [P in keyof TUP]: Func<A, TUP[P]> };
type ParamFuncTuple<A, C extends Class> = FuncTuple<A, ConstructorParameters<C>>;

function boo<C extends Class>(
  text: unknown,
  funcs: ParamFuncTuple<unknown, C>,
  c: Class
) {
  const params = funcs.map((f) => f(text))
}

Aqui o compilador diz const paramshas type any[]o que não é muito promissor. Isso é possível?

typescript
  • 1 1 respostas
  • 31 Views

1 respostas

  • Voted
  1. Best Answer
    jcalz
    2025-04-05T02:43:38+08:002025-04-05T02:43:38+08:00

    Eu sugeriria escrever a assinatura de chamada convert()assim:

    declare function convert<T, A extends any[], O extends object>(
      mapperArg: T,
      mappers: { [I in keyof A]: (mapperArg: T) => A[I] },
      ctor: new (...args: A) => O
    ): O;
    

    Existem três parâmetros de tipo genéricoT : corresponde à entrada para os mapeadores (você chamou assim, stuffmas eu nomeei mapperArgaqui); Acorresponde à tupla de tipos de parâmetros do construtor; e Ocorresponde ao tipo de instância construído.

    O importante aqui é que o tipo de mappersé um tipo de tupla mapeado sobre os índices de A; para cada índice Ido Atipo de tupla, onde A[I]é o tipo de elemento de Anaquele índice, o tipo de elemento de mappersnaquele índice é o tipo de função (mapperArg: T) => A[I].

    Vamos testar essa digitação:

    class Foo {
      a: string; b: number; c: boolean;
      constructor(a: string, b: number, c: boolean) {
        this.a = a;
        this.b = b;
        this.c = c;
      }
    }
    
    const foo = convert("hello", [s => s, s => s.length, s => s.length < 4], Foo);
    // convert<string, [a: string, b: number, c: boolean], Foo>
    // mappers: [a: (m: string) => string, b: (m: string) => number, c: (m: string) => boolean]
    // const foo: Foo;
    

    Aqui o TypeScript infere que Té string, que Oé (o tipo de instância) Fooe que Aé o tipo de tupla [a: string, b: number, c: boolean]. Portanto, o tipo de mappersé computado para ser o tipo de tupla mapeado [a: (mapperArg: string) => string, b: (mapperArg: string) => number, c: (mapperArg: string) => boolean], conforme desejado.


    Quanto à implementação de convert(), poderia ser assim:

    function convert<T, A extends any[], O extends object>(
      mapperArg: T,
      mappers: { [I in keyof A]: (mapperArg: T) => A[I] },
      ctor: new (...args: A) => O
    ): O {
      return new ctor(...(mappers.map(fn => fn(mapperArg)) as A));
    }
    

    Observe que o resultado de mappers.map(fn => fn(mapperArg))tem que ser afirmado como A; não é realmente possível para o compilador TypeScript verificar se o resultado é do tipo A, porque o sistema de tipos não é expressivo o suficiente para capturar o que Array.prototype.mapacontece com tipos de tupla arbitrários para funções de retorno de chamada arbitrárias. Veja Mapeando valor de tipo de tupla para valor de tipo de tupla diferente sem conversões . Você se deparou com isso com const paramshas type any[]; pode não ser promissor, mas também não é realmente evitável, então é razoável usar uma única afirmação de tipo e seguir em frente.

    Link do playground para o código

    • 2

relate perguntas

  • Por que usamos colchetes `[]` em condicionais?

  • Problema do Nestjs em relação aos módulos

  • Como obter metadados datilografados de um arquivo?

  • Como converter array em tipo const de retorno de objeto assim?

  • "Nenhuma sobrecarga corresponde" minha chamada Object.assign(); como posso corrigir isso?

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Reformatar números, inserindo separadores em posições fixas

    • 6 respostas
  • Marko Smith

    Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não?

    • 2 respostas
  • Marko Smith

    Problema com extensão desinstalada automaticamente do VScode (tema Material)

    • 2 respostas
  • Marko Smith

    Vue 3: Erro na criação "Identificador esperado, mas encontrado 'import'" [duplicado]

    • 1 respostas
  • Marko Smith

    Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores?

    • 1 respostas
  • Marko Smith

    Como faço para corrigir um erro MODULE_NOT_FOUND para um módulo que não importei manualmente?

    • 6 respostas
  • Marko Smith

    `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso?

    • 3 respostas
  • Marko Smith

    Um programa vazio que não faz nada em C++ precisa de um heap de 204 KB, mas não em C

    • 1 respostas
  • Marko Smith

    PowerBI atualmente quebrado com BigQuery: problema de driver Simba com atualização do Windows

    • 2 respostas
  • Marko Smith

    AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos

    • 1 respostas
  • Martin Hope
    Fantastic Mr Fox Somente o tipo copiável não é aceito na implementação std::vector do MSVC 2025-04-23 06:40:49 +0800 CST
  • Martin Hope
    Howard Hinnant Encontre o próximo dia da semana usando o cronógrafo 2025-04-21 08:30:25 +0800 CST
  • Martin Hope
    Fedor O inicializador de membro do construtor pode incluir a inicialização de outro membro? 2025-04-15 01:01:44 +0800 CST
  • Martin Hope
    Petr Filipský Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não? 2025-03-23 21:39:40 +0800 CST
  • Martin Hope
    Catskul O C++20 mudou para permitir a conversão de `type(&)[N]` de matriz de limites conhecidos para `type(&)[]` de matriz de limites desconhecidos? 2025-03-04 06:57:53 +0800 CST
  • Martin Hope
    Stefan Pochmann Como/por que {2,3,10} e {x,3,10} com x=2 são ordenados de forma diferente? 2025-01-13 23:24:07 +0800 CST
  • Martin Hope
    Chad Feller O ponto e vírgula agora é opcional em condicionais bash com [[ .. ]] na versão 5.2? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench Por que um traço duplo (--) faz com que esta cláusula MariaDB seja avaliada como verdadeira? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng Por que `dict(id=1, **{'id': 2})` às vezes gera `KeyError: 'id'` em vez de um TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos 2024-03-20 03:12:31 +0800 CST

Hot tag

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

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve