服务器之家

服务器之家 > 正文

C/C++宏替换实现详解

时间:2021-10-06 13:19     来源/作者:fater

基本形式

?
1
#define name replacement_text

通常情况下,#define 指令占一行,替换文本是 define 指令行尾部的所有剩余部分,但也可以把一个较长的宏定义分成若干行,这时需要在待续的行末尾加上一个反斜杠符 ``。

宏定义也可以带参数,这样可以对不同的宏调用使用不同的替换文本。例:

?
1
#define max(A, B) ((A) > (B) ? (A) : (B))

宏展开中的陷阱

仔细考虑一下 max 的展开式,其中的表达式会被计算两次,因此如果表达式中包含自增运算符或输入/输出等行为,则会出现不正确的情况,例如上述的宏 max:

?
1
max(i++, j++) // wrong

另外还需要注意,适当使用圆括号以保证计算次序的正确性,例如:

?
1
#define square(x) x * x // wrong

当用 square(z+1) 调用该宏定义时会出错。

#undef

在头文件 <stdio.h> 中,getchar 与 putchar 函数在实际中常常被定义为宏,这样可以避免处理字符时调用函数所需的运行时开销。<ctype.h> 头文件中定义的函数也常常是通过宏实现的。

可以通过 #define 取消名字的宏定义,这样做可以保证后续的调用是函数调用,而不是宏调用:

?
1
2
3
#undef getchar
 
int getchar(void) { ... }

宏参数、# 和 ##

如果在宏定义的替换文本中,参数名以 # 作为前缀则结果将被扩展为由实际参数替换该参数的带引号的字符串。例如,可以将它与字符串连接运算结合起来编写一个调试打印宏:

?
1
#define dprint(expr) printf(#expr " = %gn", expr)

使用语句

?
1
dprint(x/y);

调用该宏时,该宏将被扩展为:

?
1
printf("x/y" " = %gn", x/y);

其中的字符串被拼接起来了,这样,该宏调用的效果等价于

?
1
printf("x/y = %gn", x/y);

预处理器运算符 ## 为宏扩展提供了一种连接实际参数的手段。如果替换文本中的参数与 ## 相邻,则该参数将被实际参数替代,## 与前后的空白符将被删除,并对替换后的结果重新扫描。例如,下面定义的宏 paste 用于连接两个参数:

?
1
#define paste(front, back) front ## back

因此,宏调用 paste(name, 1) 的结果将建立记号 name1。

参考文献:

Brian W. Kernighan, Dennis M. Ritchie.The C Programming Language (Second Edition)[M].机械工业出版社:北京,2004:76-77.

到此这篇关于C/C++宏替换实现详解的文章就介绍到这了,更多相关C/C++宏替换内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://segmentfault.com/a/1190000038216844

标签:

相关文章

热门资讯

yue是什么意思 网络流行语yue了是什么梗
yue是什么意思 网络流行语yue了是什么梗 2020-10-11
2020微信伤感网名听哭了 让对方看到心疼的伤感网名大全
2020微信伤感网名听哭了 让对方看到心疼的伤感网名大全 2019-12-26
背刺什么意思 网络词语背刺是什么梗
背刺什么意思 网络词语背刺是什么梗 2020-05-22
2021年耽改剧名单 2021要播出的59部耽改剧列表
2021年耽改剧名单 2021要播出的59部耽改剧列表 2021-03-05
苹果12mini价格表官网报价 iPhone12mini全版本价格汇总
苹果12mini价格表官网报价 iPhone12mini全版本价格汇总 2020-11-13
返回顶部