前阵子仔细重新研究了一下C的宏展开。总结起来,有以下几个主要规则:

  1. 每次宏展开的结果会被重复扫描,直到没有任何可展开的宏为止。
  2. 每展开一个宏,都会记住这次展开,在这个宏展开的结果及其后续展开中,不再对相同的宏做展开。
  3. 带参数的宏,先对参数做展开,除非宏定义体中包含#或##
    a) #表示将后续标识符转换为字符串
    b) ##表示将两个标识符连接成一个标识符
    c) 注意参数展开的结果中即使有逗号(,),也不视为参数的分隔符
  4. 如果宏定义中带有参数,而代码中出现同样标识符时没有参数,不视为宏。

下面的三段代码分别解释了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宏展开的几个注意事项的更多相关文章

  1. C中宏展开问题

    C中宏展开问题 简单记录一下碰到的问题. #define STR(x) #x 我们知道使用上面的宏可以将x转换为字符串"x". 但是如果这样用: #define NUM 3 #de ...

  2. 一个C语言宏展开问题

    转自一个C语言宏展开问题 一个令人比较迷惑的问题,学C语言好多年,今天终于搞明白,记之. ------------------------------------------------------- ...

  3. c语言 预处理的使用 宏展开下的#,##

    1. #include   包含头文件 2.define 宏定义(可以理解为替换,不进行语法检查) 写法 #define 宏名 宏体  加括号 #define ABC (5+3) #define AB ...

  4. define的用法与注意事项

    ------------------------------------------------- 在编程使用宏替换时,当字符串中不只一个符号时,加上括号表现出优先级, 如果是带参数的宏定义,则要给宏 ...

  5. 预处理命令[#define]说明

    宏定义 宏定义是对一些常见的变量.字符串等进行定义,被定义的数据在编译会进行自动替换.有时一些变量或字符串被多次使用,当需要修改时,就需要对源文件中它们出现的地方一一修改,效率比较低,而通过宏定义,只 ...

  6. [Linux]系统调用理解(1)

    本文是Linux系统调用专栏系列文章的第一篇,对Linux系统调用的定义.基本原理.使用方法和注意事项大概作了一个介绍,以便读者对Linux系统调用建立一个大致的印象. 什么是系统调用? Linux内 ...

  7. 面试问题4:C语言预处理包括哪些

    问题描述:C语言 预处理包括哪些操作 C语言的三种预处理包括:宏定义(#define).文件包含(#include).条件编译(#if.#else.#endif). 对于宏定义的介绍: 宏定义必须写在 ...

  8. C++ Primer Plus读书笔记

    第五章 循环和关系表达式 1. 2.类别别名: (1)   #define FLOAT_POINTER float * FLOAT_POINTER pa, pb; 预处理器置换将该声明转换成  flo ...

  9. 宏定义中的##操作符和... and _ _VA_ARGS_ _

    1.Preprocessor Glue: The ## Operator 预处理连接符:##操作符 Like the # operator, the ## operator can be used i ...

随机推荐

  1. xcode8+iOS10问题

    .xcode升级到8.0后打印的问题 ()xcode8会打印一些莫名其妙的log 解决方法:Scheme里面添加OS_ACTIVITY_MODE = disable ()xcode8打印log不完整 ...

  2. IOS监听屏幕状态

    一.定义两个宏   //锁屏通知 #define NotificationOff CFSTR("com.apple.springboard.lockcomplete")   //解 ...

  3. 新版macbook air OS X El Capitan 10.11安装WIN找不到驱动介质???

    这个问题已经解决 首先进入Boot Camp6 以后 顶上会有一个操作 -下载windowns 驱动程序 保存在优盘里面.然后再分区选择ISO(中间和你前面做的一样)电脑重启进入WIN安装 到你们出现 ...

  4. 关于ř与画面的集成---- k均值聚类

    1.利用R内置数据集iris: 2.通过Rserve 包连接tableau,服务器:localhost,默认端口6311: 3.加载数据集iris: 4.编辑字段:Cluster <span s ...

  5. JQ完成表格单元格顺序的上移下调

    如有指教及疑问,欢迎留言 HTML代码 <table class="exampletable"> <thead> <tr> <th> ...

  6. ABAP开发顾问必备:SAP ABAP开发技术总结

    声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...

  7. 《BI那点儿事—数据的艺术》目录索引

    原创·<BI那点儿事—数据的艺术>教程免费发布 各位园友,大家好,我是Bobby,在学习BI和开发的项目的过程中有一些感悟和想法,整理和编写了一些学习资料,本来只是内部学习使用,但为了方便 ...

  8. DataTable转换为Json字符串的三种方法

    //第一种:使用StringBuilder  public string DataTableToJson(DataTable table) { var JsonString = new StringB ...

  9. linux下的nodejs安装

      linux下安装nodejs的方式: 1.源码安装 2.nvm安装 这里推荐使用nvm安装,避免下载nodejs源码:   安装步骤: 一.安装git        一般linux系统的git版本 ...

  10. UGUI text image 等加Shadow OutLine等

    Text,文本控件,同NGUI中的Label.支持动态字库.大小调节.富文本(基本的html标签格式)等等.描边.阴影等需要Effect组件支持.Add Component-->UI--> ...