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)