1.#在英语里叫做pound,在C语言的宏定义中,一个#表示字符串化,两个#代表concatenate。
</pre><p></p><p>2.<span style="font-family:宋体">实例一:</span></p><p></p><pre name="code" class="cpp">#include <iostream> void quit_command(){ printf("I am quit command\n"); } void help_command(){ printf("I am help command\n"); } struct command { char * name; void (*function) (void); }; #define COMMAND(NAME) {#NAME,NAME##_command} #define PRINT(NAME) printf("token"#NAME"=%d\n", token##NAME) main(){ int token9=9; PRINT(9); struct command commands[] = { COMMAND(quit), COMMAND(help), }; commands[0].function(); }
结果是:
那么下面的一个经典题目的答案呢?
3.程序如下:
#include <stdio.h> #define f(a,b) a##b #define g(a) #a #define h(a) g(a) int main(){ printf("%s\n", h(f(1,2))); printf("%s\n", g(f(1,2))); return 0;}
运行结果如下:
为什么是这个结果呢?先看关于宏展开。
4.宏展开
预处理过程的几个步骤:
1)字符集转换(如三联字符)
2)断行链接/
3)注释处理,/* comment */,被替换成空格
4)执行预处理命令,如#inlcude、#define、#pragma、#error等
5)转义字符替换
6)相邻字符串拼接
7)将预处理记号替换为词法记号
第4步即如何展开宏函数的规则:在展开当前宏函数时,如果形参有#或##则不进行宏参数的展开,否则先展开宏参数,再展开当前宏。
5.总结
综合以上,对于这道题来说,第一行h(f(1,2)),由于h(a)非#或者##所以先展开其参数f(1,2),即12,所以变成h(12),然后再宏替换为g(12),再次替换为12。
第二行g(f(1,2)),宏g(a)带有#,所以里面的f(1,2)不展开,所以变成f(1,2)