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

注意:宏定义中的#的意思:宏定义中#是“字符串化”的意思,是把跟在后面的参数转换成一个字符串,例如#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


  • 3
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值