在C中,宏的定义支持可变个数的参数,用三个点号来表示,和可变参数的函数定义是比较类似的。下面是一个例子:
#define debug(format, ...) fprintf (stderr, format, __VA_ARGS__)
其中”…”就表示可变参数,可以是0个到多个,在编译的时候__VA_ARGS__
会被替换成相应的参数列表。
debug("%d apples\n", 10);
替换的结果就是:
fprintf(stderr, "%d apples\n", 10);
然而编译下面的代码就会有问题:
debug("10 apples\n");
错误信息如下所示:
这是由于代码被扩展成了fprintf(stderr, "10 apples", )
,后面多了一个逗号,通过把##
加在__VA_ARGS__
前面就可以显式让预处理器去掉前面的逗号。
#define debug(format, ...) fprintf (stderr, format, ## __VA_ARGS__)
下面是一个完整的例子(来自msdn):
// variadic_macros.cpp
#include <stdio.h>
#define CHECK1(x, ...) if (!(x)) { printf(__VA_ARGS__); }
#define CHECK2(x, ...) if ((x)) { printf(__VA_ARGS__); }
#define CHECK3(...) { printf(__VA_ARGS__); }
#define MACRO(s, ...) printf(s, ##__VA_ARGS__)
int main() {
CHECK1(0, "here %s %s %s", "are", "some", "varargs1(1)\n");
CHECK1(1, "here %s %s %s", "are", "some", "varargs1(2)\n"); // won't print
CHECK2(0, "here %s %s %s", "are", "some", "varargs2(3)\n"); // won't print
CHECK2(1, "here %s %s %s", "are", "some", "varargs2(4)\n");
// always invokes printf in the macro
CHECK3("here %s %s %s", "are", "some", "varargs3(5)\n");
MACRO("Today is %d-%d-%d\n", 2015, 11, 28);
MACRO("hello, world\n");
return 0;
}
在终端执行gcc test.cpp && ./a.out
可以得到下面的输出:
here are some varargs1(1)
here are some varargs2(4)
here are some varargs3(5)
Today is 2015-11-28
hello, world
参考资料:
3.gcc官方文档