程序开发或者看工程代码时可能在宏定义中有涉及到#、##、__VA_ARGS__和##__VA_ARGS__的使用,简单介绍下使用方法。
1、 #–将其后面紧跟的符号转换为字符串
#操作符是将其后面紧跟的符号变为字符串的形式,示例代码:
#define PRINT_VAL(n, val) printf("%s = %d\n", #n, val)
测试:
int x1 = 3;
PRINT_VAL(x1, x1);
输出:
x1 = 3
说明:
宏定义是在编译时把使用宏的代码替换为宏定义代码,这里会将PRINT_VAL中的第一个参数x1当成字符串来处理,第二个参数取其实际的值,因此输出结果为x1 = 3。
2、 ##–将##左右两边的符号连接在一起变成一个符号
##用于将其左右两侧的操作符连接成一个操作符,可以用于定义变量和运算,示例代码:
#define PRINT_VAL(n, val) printf("%s = %d\n", #n, val)
#define VAR_NAME(n) x##n
#define SUM_CALC(num, val) x##num += val
测试:
int VAR_NAME(0) = 0;
int VAR_NAME(1) = 1;
int VAR_NAME(2) = 2;
int VAR_NAME(3) = 3;
int VAR_NAME(4) = 4;
int VAR_NAME(5) = 5;
SUM_CALC(0, 7);
PRINT_VAL(x0, x0);
输出:
x0 = 7
说明:
(1) 前面6行代码是定义了6个变量,x0~x5,SUM_CALC操作是进行加法计算,这里是x0 += 7,最后一句是打印输出。
(2) ##在将两个符号连接成一个符号时,必须满足C++的命名规则,否则程序编译报错,同时为了避免#和##操作导致的歧义,在一个宏操作中最好只存在#或者##操作,或者没有#和##号操作,如果没有指定#和##预处理操作符的计算次序,会产生问题,除非特别情形,不要一起使用#和##操作符。
(3) 通常##预操作符和__VA_ARGS__一起使用。
3、 __VA_ARGS__操作—表示该部分允许输入可变参数
类似于printf函数里的格式化输出,使用__VA_ARGS__可以使宏定义也支持可变参数的输入,这个是C99新标准中增加的内容,示例代码:
#define LOGD(format, ...) printf("debug: " format, __VA_ARGS__)
调用:
int y0 = 1;
LOGD("y0 = %d\n", y0);
说明:
(1) format为一个字符串,与“debug :”为两个字符串,中间空格隔开,C++编译器会把在字符串拼接成一个字符串后在并处理;
(2) …即为__VA_ARGS__操作符,表示程序调用时参数个数是未知的,可以在调用时添加参数
4、 ##VA_ARGS–支持可变参数的输入或者无参数输入
使用__VA_ARGS__有个问题,当不需要可变参数时,3中示例代码编译会有问题的,在最后一个参数后面会多一个“,”,上例中编译如下代码就会报错
LOGD(“test y0\n”);
##预处理操作符除了具备连接两个符号的功能,当一端没有符号时可以不做连接处理,同时去除“,”以保证程序可以编译执行。
示例代码:
#define LOGD(format, …) printf("debug: " format, ##VA_ARGS)
调用
LOGD("y0 = %d\n", y0);
LOGD("test y0\n");
输出:
debug: y0 = 1
debug: test y0
说明:
建议使用__VA_ARGS__时与##配合使用;##__VA_ARGS__在设置log输出的宏定义时比较常见。