C/C++中的宏定义及#和##的作用

在宏定义中经常看到符号###的使用,该篇博客就结合宏定义的展开方式说明这两种符号的具体作用。
本文参考学习该篇博客:参考来源——#和##的用法

用法说明

#define f(a,b) a##b      //     (1)
#define g(a)   #a        //     (2) 
#define h(a)   g(a)      //     (3)  

在上述三个宏定义中使用了符号###,使用上述宏如下代码所示

printf("%s\n", g(f(1, 2)));
printf("%s\n", h(f(1, 2)));

输出以下结果:
f(1, 2)
12

根据以上代码的输出结果,可以总结出###的用法:

符号#表示“字符串化”的意思,就是把跟在后面的参数转换成一个字符串,例如上述代码中g(f(1, 2))输出的结果为f(1, 2)
符号##是一个连接符号,用于把参数连在一起,例如a##b等同于ab

但是大家一定对于上述代码中第二输出感觉到很疑惑,想当然认为输出结果为f(1, 2),这里就需要注意以下两点:

 如果宏定义是带#的,如(2)所示,则直接替换。g(f(1,2))--->#f(1,2)--->"f(1,2)"
 如果宏定义是不带#的,如(1)、(3)所示,基本原则为展开参数然后替换。   
	步骤为:由外层向里层走,如果碰到的是以非#开头的宏,则继续往里层走,直到最里层,然后开始往外层展开。  
	如果碰到的是以#开头的宏,则不再往里层走,往外层展开。h(f(1,2))--->h(12)--->g(12)--->#12--->"12"

下面再举一些例子,大家可以在每行代码的后面看到具体的输出结果:

char a = 'c';
cout << g(a) << endl; // "a"
cout << g(g(a)) << endl; // "g(a)"
printf("%s\n", h(f(1, 2)));   // "12"
printf("%s\n", g(f(1, 2))); // "f(1,2)"
printf("%s\n", g(h(f(1, 2)))); // "h(f(1,2))"
printf("%s\n", h(g(f(1, 2)))); // ""f(1,2)""
printf("%s\n", h(h(f(1, 2)))); // ""12""
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章