首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >手动代码中包含C++20 __VA_OPT__错误的类函数宏

手动代码中包含C++20 __VA_OPT__错误的类函数宏
EN

Stack Overflow用户
提问于 2021-07-22 20:25:26
回答 1查看 149关注 0票数 2

我正在从递归宏和C++20 __VA_OPT__的伟大手册中构建代码:https://www.scs.stanford.edu/~dm/blog/va-opt.html

代码是

代码语言:javascript
复制
#include <iostream>

#define PARENS ()

// Rescan macro tokens 256 times
#define EXPAND(arg) EXPAND1(EXPAND1(EXPAND1(EXPAND1(arg))))
#define EXPAND1(arg) EXPAND2(EXPAND2(EXPAND2(EXPAND2(arg))))
#define EXPAND2(arg) EXPAND3(EXPAND3(EXPAND3(EXPAND3(arg))))
#define EXPAND3(arg) EXPAND4(EXPAND4(EXPAND4(EXPAND4(arg))))
#define EXPAND4(arg) arg

#define FOR_EACH(macro, ...)                                    \
  __VA_OPT__(EXPAND(FOR_EACH_HELPER(macro, __VA_ARGS__)))
#define FOR_EACH_HELPER(macro, a1, ...)                         \
  macro(a1)                                                     \
  __VA_OPT__(FOR_EACH_AGAIN PARENS (macro, __VA_ARGS__))
#define FOR_EACH_AGAIN() FOR_EACH_HELPER

#define ENUM_CASE(name) case name: return #name;

#define MAKE_ENUM(type, ...)                    \
enum type {                                     \
  __VA_ARGS__                                   \
};                                              \
constexpr const char *                          \
to_cstring(type _e)                             \
{                                               \
  using enum type;                              \
  switch (_e) {                                 \
  FOR_EACH(ENUM_CASE, __VA_ARGS__)              \
  default:                                      \
    return "unknown";                           \
  }                                             \
}

MAKE_ENUM(MyType, ZERO, ONE, TWO, THREE);

void
test(MyType e)
{
  std::cout << to_cstring(e) << " = " << e << std::endl;
}

int
main()
{
  test(ZERO);
  test(ONE);
  test(TWO);
  test(THREE);
}

根据https://en.cppreference.com/w/cpp/compiler_support#C.2B.2B20_features的说法,__VA_OPT__在MSVC和Clang中得到了支持,而在GCC中只得到了部分支持。

但令我惊讶的是,手册中的代码只能在GCC中运行,在其他编译器中会产生错误:

代码语言:javascript
复制
<source>(36): warning C4003: not enough arguments for function-like macro invocation 'FOR_EACH_HELPER'
<source>(36): warning C4003: not enough arguments for function-like macro invocation 'ENUM_CASE'
<source>(36): error C2059: syntax error: 'case'
<source>(36): error C2065: 'ENUM_CASE': undeclared identifier
<source>(36): error C3861: 'FOR_EACH_HELPER': identifier not found
<source>(36): error C2059: syntax error: ')'
<source>(36): error C2146: syntax error: missing ';' before identifier 'default'

https://gcc.godbolt.org/z/PreYcbz7f

这是不是意味着根据C++20,手动代码的格式不是很好,需要一些特定于GCC的语言扩展?

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2021-07-22 21:51:16

If you look here,我增加了一个__VA_OPT__支持检测器。

代码语言:javascript
复制
#define PP_THIRD_ARG(a,b,c,...) c
#define VA_OPT_SUPPORTED_I(...) PP_THIRD_ARG(__VA_OPT__(,),true,false,)
#define VA_OPT_SUPPORTED VA_OPT_SUPPORTED_I(?)

static_assert(VA_OPT_SUPPORTED);

gcc和clang通过了它;MSVC版本不能。

然后我查看了clang的警告;这是一个警告,如果您向...传递0个参数,它会告诉您。

它看起来是假的。我用-Wno-gnu-zero-variadic-macro-arguments禁用了它,然后你的代码就可以正常工作了。

所以不管是什么版本的MSVC在龙箭上似乎都不支持__VA_OPT__,而gcc和clang版本支持。并且该clang版本在您选择的警告集中有一个虚假警告。

To fix MSVC, pass /Zc:preprocessor。默认情况下,它使用传统的预处理器。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/68484818

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档