#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)));
注意:宏定义中的#的意思:宏定义中#是“字符串化”的意思,是把跟在后面的参数转换成一个字符串,例如#a等同于“a”,#abc等同于“abc”。宏定义中的##是一个连接符号,用于把参数连在一起。例如,a##b##c等同于abc
计算方法:一、如果宏定义是带#的,如(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""
宏解析
1. ##操作符
##操作符它的作用是在替代表中将其前后的参数连接成为一个预处理符号,它不能出现于宏替代表的开端和末尾。
例:
#define concat(s,t) s##t
#define AAA ABC
concat(A, AA)
将被替换成
ABC
2. 重新扫描和替换
在替换列表中的所有参数替换过之后,预处理器将对结果token序列重新扫描以便对其中的宏再次替换。
当正在替换的宏在其替换列表中发现自身时,就不再对其进行替换。今儿,在任何正在嵌套替换的宏的替换过程中遇到正被替换的宏就对其不再进行替换(防止递归)。
例:
#define ROOT AAA CCC
#define AAA ROOT
ROOT
将被替换成
ROOT CCC