#的功能是将其后面的宏参数进行字符串化操作(Stringfication),简单说就是在对它所引用的宏变量通过替换后在其左右各加上一个双引号。比如下面代码中的宏:
#define WARN_IF(EXP)
do{ if (EXP)
fprintf(stderr, "Warning: " #EXP " "); }
while(0)
那么实际使用中会出现下面所示的替换过程:
WARN_IF (divider == 0);
被替换为
do {
if (divider == 0)
fprintf(stderr, "Warning" "divider == 0" " ");
} while(0);
这样每次divider(除数)为0的时候便会在标准错误流上输出一个提示信息。
当宏参数是另一个宏时
需要注意的是凡宏定义里有用'#'或'##'的地方宏参数是不会再展开.
2, 当有'#'或'##'的时候
#define A (2)
#define STR(s) #s
#define CONS(a,b) int(a##e##b)
printf("int max: %s
", STR(INT_MAX)); // INT_MAX #include<climits>
这行会被展开为:
printf("int max: %s
", "INT_MAX");
printf("%s
", CONS(A, A)); // compile error
这一行则是:
printf("%s
", int(AeA));
INT_MAX和A都不会再被展开(宏定义里有用'#'或'##'), 然而解决这个问题的方法很简单. 加多一层中间转换宏.
加这层宏的用意是把所有宏的参数在这层里全部展开, 那么在转换宏里的那一个宏(_STR)就能得到正确的宏参数.
#define A (2)
#define _STR(s) #s
#define STR(s) _STR(s) // 中间转换宏
#define _CONS(a,b) int(a##e##b)
#define CONS(a,b) _CONS(a,b) // 中间转换宏
printf("int max: %s
", STR(INT_MAX)); // INT_MAX,int型的最大值,为一个变量 ,#include<climits>
输出为: int max: 0x7fffffff
STR(INT_MAX) --> _STR(0x7fffffff) 然后再转换成字符串;
printf("%d
", CONS(A, A));
输出为:200
展开过程如下
CONS(A, A) --> _CONS((2), (2)) --> int((2)e(2))