C宏展开的几个注意事项
前阵子仔细重新研究了一下C的宏展开。总结起来,有以下几个主要规则:
- 每次宏展开的结果会被重复扫描,直到没有任何可展开的宏为止。
- 每展开一个宏,都会记住这次展开,在这个宏展开的结果及其后续展开中,不再对相同的宏做展开。
- 带参数的宏,先对参数做展开,除非宏定义体中包含#或##
a) #表示将后续标识符转换为字符串
b) ##表示将两个标识符连接成一个标识符
c) 注意参数展开的结果中即使有逗号(,),也不视为参数的分隔符 - 如果宏定义中带有参数,而代码中出现同样标识符时没有参数,不视为宏。
下面的三段代码分别解释了2, 3, 4. 注释中描述了宏每一步展开的细节
这段代码主要解释规则2.(~表示已经被展开过)
#define foo foo bar
#define bar bar bar foo
#define foo2(a) bar2(a,foo2) foo2(a) (a)
#define bar2(a, b) foo2(a) bar2(a,b) (a) (b)
foo
// |-> foo bar
// | |~ |-> bar bar foo
// |-> foo bar bar foo (至此,所有符号都已展开过)
bar
// |-> bar bar foo
// | |~ |~ |-> foo bar
// |-> bar bar foo bar (至此,所有符号都已展开过)
foo bar
// foo bar
// | |
// |-> foo bar bar bar foo
// | |~ | |~ |~ |
// |-> foo bar bar foo bar bar foo bar
foo2(1)
// |-> bar2(1, foo2) foo2(1) (1)
// | | |~
// |-> foo2(1) bar2(1, foo2) (1) (foo2) foo2(1) (1)
bar2(1, 1)
// |-> foo2(1) bar2(1,1) (1) (1)
// | | |~
// |-> bar2(1,foo2) foo2(1) (1) bar2(1,1) (1) (1)
这段代码主要解释规则3.
#define foo vfoo
#define bar vbar
#define foo2(a) #a
#define foo3(a, b) a ## _ ## b
#define foo20(a) foo2(a)
#define foo30(a, b) foo3(a, b)
#define foo40(x) foo30(x)
#define x x1,x2
foo2(foo)
// -> "foo" ('#'阻止了参数展开,如果需要展开参数,定义另一个宏,见foo20)
foo3(foo, bar)
// -> foo_bar ('##'阻止了参数展开,如果需要展开参数,定义另一个宏,见foo30)
foo20(foo)
// | |-->vfoo (foo20带有一个参数,匹配宏定义,先展开参数)
// | |
// |-> foo2(vfoo)
// |-> "vfoo"
foo30(foo, bar)
// | |-->vfoo, |->vbar
// | | |
// |-> foo3(vfoo, vbar)
// |-> vfoo_vbar
foo30(x)
// 错误,参数个数不匹配, x中的逗号不视为参数分隔符。如果需要将这个逗号作为分隔符,定义另一个宏,见foo40
foo40(x)
// | |-> x1,x2
// |-> foo30(x1,x2) //这个时候,逗号视为合法分隔符。
// |-> foo3(x1,x2)
// |-> x1_x2
这段代码主要解释规则4.
#define foo(a) foo=a(x)
#define bar(a) (bar=a)
#define foo2(a) foo=a(x,y)
#define bar2(a,b) bar(a*b)
foo
// 参数个数不匹配,不认为是宏
bar
// 同上
foo(bar)
// |-> foo=bar(x) (bar无参数,不认为是宏)
// |-> foo=(bar=x) (此次扫描,bar符合宏定义)
foo2(bar2)
// |-> foo=bar2(x,y) (bar2无参数,不认为是宏)
// |-> foo=bar(x*y) (此次扫描,bar符合宏定义)
// |-> foo=(bar=x*y)
C宏展开的几个注意事项的更多相关文章
- C中宏展开问题
C中宏展开问题 简单记录一下碰到的问题. #define STR(x) #x 我们知道使用上面的宏可以将x转换为字符串"x". 但是如果这样用: #define NUM 3 #de ...
- 一个C语言宏展开问题
转自一个C语言宏展开问题 一个令人比较迷惑的问题,学C语言好多年,今天终于搞明白,记之. ------------------------------------------------------- ...
- c语言 预处理的使用 宏展开下的#,##
1. #include 包含头文件 2.define 宏定义(可以理解为替换,不进行语法检查) 写法 #define 宏名 宏体 加括号 #define ABC (5+3) #define AB ...
- define的用法与注意事项
------------------------------------------------- 在编程使用宏替换时,当字符串中不只一个符号时,加上括号表现出优先级, 如果是带参数的宏定义,则要给宏 ...
- 预处理命令[#define]说明
宏定义 宏定义是对一些常见的变量.字符串等进行定义,被定义的数据在编译会进行自动替换.有时一些变量或字符串被多次使用,当需要修改时,就需要对源文件中它们出现的地方一一修改,效率比较低,而通过宏定义,只 ...
- [Linux]系统调用理解(1)
本文是Linux系统调用专栏系列文章的第一篇,对Linux系统调用的定义.基本原理.使用方法和注意事项大概作了一个介绍,以便读者对Linux系统调用建立一个大致的印象. 什么是系统调用? Linux内 ...
- 面试问题4:C语言预处理包括哪些
问题描述:C语言 预处理包括哪些操作 C语言的三种预处理包括:宏定义(#define).文件包含(#include).条件编译(#if.#else.#endif). 对于宏定义的介绍: 宏定义必须写在 ...
- C++ Primer Plus读书笔记
第五章 循环和关系表达式 1. 2.类别别名: (1) #define FLOAT_POINTER float * FLOAT_POINTER pa, pb; 预处理器置换将该声明转换成 flo ...
- 宏定义中的##操作符和... and _ _VA_ARGS_ _
1.Preprocessor Glue: The ## Operator 预处理连接符:##操作符 Like the # operator, the ## operator can be used i ...
随机推荐
- XArp汉化破解专业版,强大易用的ARP欺骗检测器
汉化作者:Bluefish 破解来自:http://www.52pojie.cn/thread-464808-1-1.html官方网站:http://www.xarp.net/ ----------- ...
- java 垃圾回收
转自:http://www.360doc.com/content/13/0305/10/15643_269388816.shtml
- 输入5至10之间的数字(用javaScript实现判断)
输入5至10之间的数字 ----用javaScript实现判断 代码如下: <!DOCTYPE html><html><body> <script>fu ...
- JS判断日期是否在同一个星期内,和同一个月内
今天要用到判断日期是否在同一个星期内和是否在同一个月内,在网上找了好一会儿也没找到合适的,然后自己写了一个方法来处理这个问题,思路就不详细介绍了,直接附上代码,自己测试了一下 没有问题,若有问题请在评 ...
- dom4j的quickstart
我所理解的dom4j就是用来解析XML文档的,XML文档的重要性不言而喻,用过框架的人谁不知道呢,是不是.但是实际上需要我们自己来解析XML文档的应用场景感觉不是很多,毕竟该解析的XML都已经被框架很 ...
- ElasticSearch与Spring Boot集成问题
1.None of the configured nodes are available 或者 org.elasticsearch.transport.RemoteTransportException ...
- vc6.0如何显示行号以及出现版本不兼容问题
有时编译时,提示某某行有错,但是要定位到某一行的话,如果在编辑页面能够将行号显示出来,查找也就更方便了,下面我来介绍一下让VC6.0显示行号的方法. 工具/原料 VC6.0.显示行号的插件 方 ...
- 虚拟机EAL: Error reading from file descriptor
这个是虚拟机安装固有的BUG,代码差异如下: diff --git a/lib/librte_eal/linuxapp/igb_uio/igb_uio.c b/lib/librte_eal/linux ...
- centos下安装yaf框架
安装好php环境之后 安装扩展包 $yum install php-devel /usr/bin/ 就会出现phpize工具包 下载yaf-2.2.8.gz源文件,解压后,进入源文件 phpize [ ...
- spring 声明式事务管理
简单理解事务: 比如你去ATM机取5000块钱,大体有两个步骤:首先输入密码金额,银行卡扣掉5000元钱:然后ATM出5000元钱.这两个步骤必须是要么都执行要么都不执行.如果银行卡扣除了5000块但 ...