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 / 79560023
Accepted
Dess
Dess
Asked: 2025-04-07 21:37:32 +0800 CST2025-04-07 21:37:32 +0800 CST 2025-04-07 21:37:32 +0800 CST

Pular linha na tabela x-macro

  • 772

Acho que isso não é possível, mas pergunto mesmo assim.

Dada uma macro x como a seguinte:

#define X_TABLE(_)\
  _(1, a, "hi")\
  _(2, b, "hello")\
  _(3, c, "sup")

// ...

#define X(_1, _2, _3) SomeFunc(_1, _2, _3);
X_TABLE(X)
#undef X

O acima geraria:

SomeFunc(1, a, "hi");
SomeFunc(2, b, "hello");
SomeFunc(3, c, "sup");

Existe uma maneira de pular algumas linhas de X_TABLEna avaliação de X, de modo que SomeFuncnão seja gerado para essa linha?

Talvez, gere algo assim:

SomeFunc(1, a, "hi");
SomeFunc(3, c, "sup");
c++
  • 2 2 respostas
  • 98 Views

2 respostas

  • Voted
  1. Best Answer
    Scheff's Cat
    2025-04-08T14:44:24+08:002025-04-08T14:44:24+08:00

    Isenção de responsabilidade:

    Macros X são uma técnica antiga introduzida na programação C.

    Como as macros são independentes de namespaces e escopos, elas podem ser responsáveis ​​por efeitos estranhos e inesperados em C++. (Em C, esse problema não existe devido à ausência de classes e namespaces.)

    No entanto, (pelo menos até o C++20) há alguns casos de uso que ainda não podem ser cobertos por nada além de macros — ou seja, técnicas que se baseiam no operador stringify.

    Macros X condicionais:

    Em relação ao uso condicional de linhas em macros X por OPs, a seguinte solução é possível: a introdução de uma ou mais colunas adicionais com condições. Essas colunas adicionais são usadas (em combinação com a colagem de tokens) para criar nomes de macros. Portanto, algumas linhas podem (ou não) ser usadas para expandir para um texto diferente de outras.

    Um exemplo para ilustrar isso:

    #include <iostream>
    
    #define X_TABLE(_) \
      _(1, 'a', "hi",    A) \
      _(2, 'b', "hello", B) \
      _(3, 'c', "sup",   A)
    
    void funcEvery(int i, char c, const char* text)
    {
      std::cout << "i: " << i << ", c: " << c << ", text: " << text << '\n';
    }
    
    void funcSel(int i, char c, const char* text)
    {
      std::cout << "i: " << i << ", c: " << c << ", text: " << text << '\n';
    }
    
    #define DEBUG(...) std::cout << "  " << #__VA_ARGS__ << ";\n"; __VA_ARGS__ 
    
    int main()
    {
      std::cout << "Unconditional Expanding:\n";
    #define X(_1, _2, _3, _4) DEBUG(funcEvery(_1, _2, _3)); // just not using _4
      X_TABLE(X)
    #undef X
    
      std::cout << "\nConditional Expanding:\n";
    #define X_A(_1, _2, _3) DEBUG(funcSel(_1, _2, _3)); // expand to something
    #define X_B(_1, _2, _3) // expand to something different -> e.g. nothing
    #define X(_1, _2, _3, _4) X_##_4(_1, _2, _3) // _4 for cond. expand
      X_TABLE(X)
    #undef X
    #undef X_A
    #undef X_B
    }
    

    Saída:

    Unconditional Expanding:
      funcEvery(1, 'a', "hi");
    i: 1, c: a, text: hi
      funcEvery(2, 'b', "hello");
    i: 2, c: b, text: hello
      funcEvery(3, 'c', "sup");
    i: 3, c: c, text: sup
    
    Conditional Expanding:
      funcSel(1, 'a', "hi");
    i: 1, c: a, text: hi
      funcSel(3, 'c', "sup");
    i: 3, c: c, text: sup
    

    Demonstração no coliru

    • 3
  2. Lundin
    2025-04-08T15:05:51+08:002025-04-08T15:05:51+08:00

    Versões mais recentes dos padrões C++ e C suportam __VA_OPT__. Você pode usar isso para introduzir um quarto token pp à lista, habilitando um comportamento especial:

    #define X_TABLE(_)\
      _(1, a, "hi", something)\
      _(2, b, "hello")\
      _(3, c, "sup", something)
    
    #define X(_1, _2, _3, ...) __VA_OPT__( SomeFunc(_1, _2, _3); )
    X_TABLE(X)
    #undef X
    

    Pré-processado:

    SomeFunc(1, a, "hi"); 
    SomeFunc(3, c, "sup");
    
    • 2

relate perguntas

  • Por que os compiladores perdem a vetorização aqui?

  • Erro de compilação usando CMake com biblioteca [fechada]

  • Erro lançado toda vez que tento executar o premake

  • Como criar um tipo de octeto semelhante a std::byte em C++?

  • Somente operações bit a bit para std::byte em C++ 17?

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