【C语言】关于宏定义中#和##符号的使用和宏定义展开问题

时间:2022-09-23 14:24:19

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();
}

结果是:

 【C语言】关于宏定义中#和##符号的使用和宏定义展开问题

那么下面的一个经典题目的答案呢?

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;}


运行结果如下:

【C语言】关于宏定义中#和##符号的使用和宏定义展开问题

为什么是这个结果呢?先看关于宏展开。

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)