带参数的宏的定义如下:
#define 标识符(标识符表) 记号序列
例如:#define MAX(a,b) ((a)>(b)?(a):(b))
MAX(5,2)就是一个宏调用,宏调用的实际参数是用逗号分隔的记号序列,用引号或嵌套的括号括起来的逗号不能用于分隔实参,在分隔的过程中,实参不进行宏扩展。宏调用时,实际参数的数目必须与定义中的形式参数的数目匹配,实参被分离后,前导和尾部的空格字符将被删除,然后用实参替换未用引号引起来的相应形参的标识符。除非替换序列中的形参前面有一个#运算符,或者其前面或后面有一个##运算符,否则,在插入前要对宏调用的实际参数记号进行检查,并在必要的时候进行扩展。
关于#和##在C语言的宏中,#的功能是将其后面的宏参数进行字符串化操作,简单说就是在对它所引用的宏变量通过替换后在其左右各加上一个双引号。##被称为连接符,用来将两个Token连接为一个Token。注意这里连接的对象是Token就行,而不一定是宏的变量。
下面有#和##运算符用于宏定义中的例子。
#include <stdio.h>
#define TO_STRING2(x) #x
#define TO_STRING(x) TO_STRING1(x)
#define TO_STRING1(x) #x
#define PARAM(x) #x
#define ADDPARAM(x) INT_##x
int main()
{
const char *str = TO_STRING(PARAM(ADDPARAM(1)));
printf("%s\n",str);
str = TO_STRING2(PARAM(ADDPARAM(1)));
printf("%s\n",str);
return 0;
}
运行结果如下:
"ADDPARAM(1)"
PARAM(ADDPARAM(1))
#include <stdio.h>
#define TO_STRING2(x) a_##x
#define TO_STRING(x) TO_STRING1(x)
#define TO_STRING1(x) #x
#define PARAM(x) #x
#define ADDPARAM(x) INT_##x
int main()
{
const char *str = TO_STRING(TO_STRING2(PARAM(ADDPARAM(1))));
printf("%s\n",str);
return 0;
}
vs编译输出如下:
a_PARAM(ADDPARAM(1))
GCC编译输出如下:
a_PARAM(INT_1)
#include <stdio.h>
#define TO_STRING2(x) a_##x
#define TO_STRING(x) TO_STRING1(x)
#define TO_STRING1(x) T(x)
#define T(x) #x
#define PARAM(x) #x
#define ADDPARAM(x) INT_##x
int main()
{
const char *str = TO_STRING(TO_STRING2(PARAM(ADDPARAM(1))));
printf("%s\n",str);
return 0;
}
运行结果如下:
a_PARAM(INT_1)
#include <stdio.h>
int ival = 0;
#define A(x) printf("%d\n", ival+=1);
#define B(x) printf("%d\n", ival+=2);
#define C() printf("%d\n", ival+=3);
int main()
{
A(B(C()));
printf("%d\n", ival);// ?, 1
return 0;
}
运行结果如下:
1
1
整理自:
1、C程序设计语言(K&R),中文版第二版,P207-208
2、http://blog.csdn.net/delphiwcdj/article/details/7040247