在GNU C中,从C99开始,宏可以接受可变数目的参数,就象可变参数函数一样。和函数一样,宏也用三个点…来表示可变参数
__VA_ARGS__ 宏用来表示可变参数的内容,简单来说就是将左边宏中 … 的内容原样抄写在右边__VA_ARGS__ 所在的位置。如下例代码:
#include <stdio.h> #define debug(...) printf(__VA_ARGS__) int main(void) { int year = 2018; debug("this year is %d\n", year); //效果同printf("this year is %d\n", year); }另外,通过一些语法,你可以给可变参数起一个名字,而不是使用__VA_ARGS__ ,如下例中的args:
#include <stdio.h> #define debug(format, args...) printf(format, args) int main(void) { int year = 2018; debug("this year is %d\n", year); //效果同printf("this year is %d\n", year); }与可变参数函数不同的是,可变参数宏中的可变参数必须至少有一个参数传入,不然会报错,为了解决这个问题,需要一个特殊的“##”操作,如果可变参数被忽略或为空,“##”操作将使预处理器(preprocessor)去除掉它前面的那个逗号。如下例所示
#include <stdio.h> #define debug(format, args...) printf(format, ##args) int main(void) { int year = 2018; debug("hello, world"); //只有format参数,没有args可变参数 }举个例子:宏定义为#define XNAME(n) x##n,代码为:XNAME(4),则在预编译时,宏发现XNAME(4)与XNAME(n)匹配,则令 n 为 4,然后将右边的n的内容也变为4,然后将整个XNAME(4)替换为 x##n,亦即 x4,故最终结果为 XNAME(4) 变为 x4。如下例所示:
#include <stdio.h> #define XNAME(n) x##n #define PRINT_XN(n) printf("x" #n " = %d\n", x##n); int main(void) { int XNAME(1) = 14; // becomes int x1 = 14; int XNAME(2) = 20; // becomes int x2 = 20; PRINT_XN(1); // becomes printf("x1 = %d\n", x1); PRINT_XN(2); // becomes printf("x2 = %d\n", x2); return 0; }