- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
如果我尝试编译以下代码:
template <typename... TArgs>
void Dummy(const TArgs &...args)
{
}
#define DUMMY(...) Dummy("Hello", ##__VA_ARGS__)
int main()
{
DUMMY();
}
我得到以下编译错误:
g++ -std=c++17 -O3 -Wall main.cpp && ./a.out
main.cpp: In function 'int main()':
main.cpp:6:48: error: expected primary-expression before ')' token
6 | #define DUMMY(...) Dummy("Hello", ##__VA_ARGS__)
| ^
main.cpp:10:5: note: in expansion of macro 'DUMMY'
10 | DUMMY();
| ^~~~~
https://coliru.stacked-crooked.com/a/c9217ba86e7d24bd
当我至少添加一个参数时,代码可以正常编译:
template <typename... TArgs>
void Dummy(const TArgs &...args)
{
}
#define DUMMY(dummy, ...) Dummy(dummy, ##__VA_ARGS__)
int main()
{
DUMMY(); // This is strange. Why does this compile?
DUMMY(1);
DUMMY(1, 2);
DUMMY(1, 2, 3);
}
https://coliru.stacked-crooked.com/a/e30e14810d70f482
但我不确定它是否正确,因为 DUMMY
至少需要一个参数,但我传递了零。
最佳答案
关于 C/C++ 宏的一个重要事实是,不带参数调用它们是不可能的,因为宏参数允许是空标记序列。
因此,DUMMY()
使用单个空参数而不是零参数调用宏 DUMMY
。这解释了为什么第二个示例有效,也解释了为什么第一个示例产生语法错误。
当 __VA_ARGS__
没有元素时,GCC 扩展从 , ##__VA_ARGS__
中删除逗号。但是一个空参数与没有参数是不一样的。当您将 DUMMY
定义为 #define DUMMY(...)
时,您保证 __VA_ARGS__
至少有一个参数,因此 ,
不会被删除。
***注意:如果您没有使用 --std
选项指定某些 ISO 标准,GCC 会对该规则做出异常(exception)处理。在这种情况下,如果 ...
是唯一的宏参数并且调用有一个空参数,那么 ,##__VA_ARGS__
确实 删除了逗号。这在 CPP manual in the Variadic Marcos section 中有说明。 :
The above explanation is ambiguous about the case where the only macro parameter is a variable arguments parameter, as it is meaningless to try to distinguish whether no argument at all is an empty argument or a missing argument. CPP retains the comma when conforming to a specific C standard. Otherwise the comma is dropped as an extension to the standard.
当 DUMMY
为 #define DUMMY(x, ...)
时,如果 DUMMY
__VA_ARGS
将为空> 仅使用一个参数调用,其中包括调用 DUMMY()
(一个空参数)和 DUMMY(0)
(一个参数,0
)。请注意,C++20 之前的标准 C 和 C++ 不允许此调用;他们要求至少有一个(可能为空的)参数对应于省略号。不过,GCC 从未施加此限制,并且无论 --std
设置如何,GCC 都会忽略 ,##__VA_ARGS__
的逗号。
从 C++20 开始,您可以使用 __VA_OPT__
内置宏作为处理逗号(以及任何其他可能需要删除的标点符号)的更标准方式。 __VA_OPT__
也避免了上面出现的空参数问题,因为它使用了不同的标准:__VA_OPT__(x)
扩展为 x
if __VA_ARGS__
包含至少一个标记;否则,它扩展为一个空序列。因此,对于这个问题中的宏,__VA_OPT__
将按预期工作。
我相信所有主要编译器现在都实现了 __VA_OPT__
,至少在它们的最新版本中是这样。
关于c++ - 即使使用 ##__VA_ARGS__ 也无法编译具有零参数的可变参数宏,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63740056/
为什么禁用类型像 type t = A of int | B of string * mutable int 虽然允许此类类型: type t = A of int | B of string * i
我正在寻找一种类似结构的数据结构,我可以从中创建多个实例并具有某种类型提示而不是不可变的。 所以我有这样的东西: class ConnectionConfig(NamedTuple): nam
我需要转到引用的结构: class SearchKnot { var isWord : Bool = false var text : String = "" var to
如sec 10.4.3中所述 当控制进入执行时,执行以下步骤 功能对象F(调用者)中包含的功能代码的上下文 提供thisArg,而调用方提供argumentsList: 如
i make a game that start display Activity indicator And activity indicator bottom display UiLable wi
编辑:我在这里不断获得支持。只是为了记录,我认为这不再重要。自从我发布它以来我就不再需要它了。 我想在 Scala 中执行以下操作... def save(srcPath: String, destP
使用可变对象作为 Hashmap 键是一种不好的做法吗?当您尝试使用已修改足以更改其哈希码的键从 HashMap 中检索值时,会发生什么? 例如,给定 class Key { int a; /
如果您在Kotlin中访问List类型的Java值,则将获得(Mutable)List!类型。 例如。: Java代码: public class Example { public stati
我编写了 str 类(内置)的以下扩展,以便执行以下操作:假设我有字符串 "Ciao" ,通过做"Ciao" - "a"我想要的结果是字符串 "Cio" 。这是执行此操作的代码,并且运行良好: cla
使用可变对象作为 Hashmap 键是一种不好的做法吗?当您尝试使用已修改足以更改其哈希码的键从 HashMap 中检索值时,会发生什么? 例如,给定 class Key { int a; /
我正在为我的公司设计一个数据库来管理商业贷款。每笔贷款都可以有担保人,可以是个人或公司,在借款业务失败时作为财务支持。 我有 3 个表:Loan、Person 和 Company,它们存储明显的信息。
我使用二进制序列化从 C# 类中保存 F# 记录。一切正常: F#: type GameState = { LevelStatus : LevelStatus
import javax.swing.JOptionPane; public class HW { public static void main(String[] args) { Strin
使用 flatbuffer mutable 有多少性能损失? 是否“正确”使用 FlatBuffers 来拥有一个应该可编辑的对象/结构(即游戏状态) 在我的示例中,我现在有以下类: class Ga
std::function create_function (args...) { int x = initial_value (args...); return [x] () mut
我需要在 for 循环中找到用户输入的字符。我通常会这样做 如果(句子[i] == 'e') 但是因为在这里,'e' 将是一个单字母字符变量,我不知道如何获取要比较的值。我不能只输入 if (sent
我有一个这样的算法: let seed: Foo = ... let mut stack: Vec = Vec::new(); stack.push(&seed); while let Some(ne
这个问题可能看起来非常基础,但我很难弄清楚如何做。我有一个整数,我需要使用 for 循环来循环整数次。 首先,我尝试了—— fn main() { let number = 10; // An
如果我有以下结构: struct MyStruct { tuple: (i32, i32) }; 以及以下函数: // This will not compile fn function(&mut s
我希望在每个 session 的基础上指定列的默认值。下面的脚本不起作用,但描述了我想如何使用它。我目前使用的是 MySQL 5.5.28,但如果需要可以升级。 CREATE TABLE my_tbl
我是一名优秀的程序员,十分优秀!