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