宏参数(Arguments)的扩展
宏分为两种,一种是 object-like 宏,比如:
#define STR "Hello, World!"
另一种是 function-like 宏,比如:
#define MIN(X, Y) ((X) < (Y) ? (X) : (Y))
对于 function-like 宏,定义时的参数叫 Parameters,比如上面宏 MIN 的参数 X、Y,当调用时,传递的参数叫 Arguments。
宏参数 Arguments
当给宏传递参数 Arguments 时,可以不用全部传递,比如:
MIN(a, b) -> ((a) < (b) ? (a) : (b)) // 完全传递 MIN(, b) -> (() < (b) ? () : (b)) // 只传递后一个 MIN(a, ) -> ((a) < () ? (a) : ()) // 只传递前一个 MIN(,) -> (() < () ? () :()) // 这种允许 MIN(, ,) // 报错,接收2个参数,传递了3个 MIN() // 报错,接收2个参数,传递了1个
宏参数 Arguments 的扩展
当向宏中传递参数 Arguments 时,在参数替换到宏里面之前,首先要对参数 Arguments 进行完全的扩展,当参数扩展完毕之后,才将最终扩展的结果替换到宏里面,同时再对整个宏进行扩展。比如调用宏 MIN(MIN(a, b), c),那么首先第一个参数 MIN(a, b)要进行扩展,扩展的结果为 ((a) < (b) ? (a) : (b)),由于第2个参数 c 不用扩展,也就是第1步得到的扩展结果为:
MIN(((a) < (b) ? (a) : (b)), c)
接下来进行第2步扩展,得到的结果为:
((((a) < (b) ? (a) : (b))) < (c) ? (((a) < (b) ? (a) : (b))) : (c))
这样做的一个好处是为了防止宏的嵌套调用,比如有下面一个宏定义:
#define f(x) x
如果有如下调用 f(f(1)),假设第一步不将参数 f(1) 完全展开,就会出现 f(f(1)) 被展开为 f(1),由于间接的自引用,宏不再继续展开,此时得到的结果就为 f(1)。而有了2步扩展,第一步参数 f(1) 被扩展成 1,然后替换后继续扩展,可以得到结果就是1。
自引用宏
一个宏在定义时,如果宏名出现在了宏定义中,那么就是一个自引用宏,比如:
#define foo (4 + foo)
如果调用这个宏 foo,按照上面两步展开的过程,那么将会是一个无限循环的过程。首先 foo 展开为 (4 + foo),然后 (4 + foo) 中的 foo 继续展开成 (4 + foo) 成为 (4 + (4 + foo)),如此继续下去。但是实际上,对于自引用宏,只扩展1次,后面不会再做扩展,也就是说调用宏 foo,最终的扩展结果就是 (4 + foo)。
对于自引用宏的一种特殊情况就是间接自引用,比如有如下宏定义:
#define x (4 + y)
#define y (2 * x)
当调用宏 x 时,首先扩展为 (4 + y),然后对 y 进行扩展,结果为 (4 + (2 * x)),如果此时继续对 x 进行扩展,那么就会无限递归,为了处理这种情况,扩展会在这一步终止,也就是最终结果是 (4 + (2 * x))。
当调用宏 y 时一样,首先扩展为 (2 * x),然后对 x 进行扩展,结果为 (2 * (4 + y)),扩展到这一步也会停止。
参数 Arguments 有字符串化(#)和连接(##)操作
在宏扩展过程当中,当参数 Arguments 有字符串化(#)和连接操作(##)时,参数的扩展就没有第1步。比如有下面宏定义:
#define AFTERX(x) X_ ## x
#define XAFTERX(x) AFTERX(x)
#define TABLESIZE 1024
#define BUFSIZE TABLESIZE
当调用 AFTERX(BUFSIZE) 时,由于参数 BUFSIZE 有连接操作(##),那么它就不会继续扩展成 TABLESIZE,TABLESIZE 继续扩展成1024,而是直接使用,也就是最终的扩展结果是 X_BUFSIZE,而不是 X_10。如果想要得到 X_10 的结果,需要做1次间接宏调用,重新定一个宏 XAFTERX 来调用宏 AFTERX,当调用宏 XAFTERX(BUFSIZE) 时,就会得到结果 X_10。
宏参数(Arguments)的扩展的更多相关文章
- js参数arguments的理解
原文地址:js参数arguments的理解 对于函数的参数而言,如下例子 function say(name, msg){ alert(name + 'say' + msg); } say('xiao ...
- C/C++语言中#的神奇作用:把宏参数字符串化/贴合宏参数
宏中"#"和"##"的用法 一.一般用法 我们使用#把宏参数变为一个字符串,用##把两个宏参数贴合在一起. #define STR(s) #s # ...
- C/C++中 # 的神奇作用:把宏参数字符串化/贴合宏参数
一.一般用法 我们使用#把宏参数变为一个字符串,用##把两个宏参数贴合在一起. #define STR(s) #s #define CONS(a,b) int(a##e##b) printf(ST ...
- [译]Javascript 参数(arguments)对象
本文翻译youtube上的up主kudvenkat的javascript tutorial播放单 源地址在此: https://www.youtube.com/watch?v=PMsVM7rjupU& ...
- 函数的默认值与动态参数arguments的总结
在js函数与作用域,了解函数基本概念中,我们发现当函数的实参有一个没有上传的时候,对应的形参time展示的值就是undefined,如下代码所示: <!DOCTYPE html> < ...
- C语言:宏参数的字符串化和宏参数的连接
在宏定义中,有时还会用到#和##两个符号,它们能够对宏参数进行操作. # 的用法 #用来将宏参数转换为字符串,也就是在宏参数的开头和末尾添加引号.例如有如下宏定义: #define STR(s) #s ...
- 隐式参数arguments
类数组对象中(长得像一个数组,本质上是一个对象):arguments 常见的对arguments的操作是三个 获取参数的长度 arguments.length 根据索引值获取某一个参数 argume ...
- Confluence 6.15 博客页面(Blog Posts)宏参数
参数是让你可以用来控制宏的格式和输出的选项.在 Confluence 存储格式或者 Wiki 标记(wikimarkup)中使用的参数名与在宏浏览器中使用的标签名是不同的,在下面我们将会用括号列出 ...
- js的隐含参数(arguments,callee,caller)使用方法
在提到上述的概念之前,首先想说说javascript中函数的隐含参数: arguments arguments 该对象代表正在执行的函数和调用它的函数的参数.[function.]arguments[ ...
随机推荐
- 循环retian
1.循环retian基本概念 循环retain的场景 比如A对象retain了B对象,B对象retain了A对象 循环retain的弊端 这样会导致A对象和B对象永远无法释放 循环retain的解决方 ...
- iOS block的用法 by -- 周傅琦君
X.1 初探Block X.1.1 宣告和使用Block 我们使用「^」运算子来宣告一个block变数,而且在block的定义最后面要加上「;」来表示一个完整的述句(也就是将整个block定义视为前面 ...
- iOS 七大手势之轻拍,长按,旋转手势识别器方法-赵小波
一.监听触摸事件的做法 如果想监听一个view上面的触摸事件,之前的做法通常是:先自定义一个view,然后再实现view的touches方法,在方法内部实现具体处理代码 通过touches方法监听vi ...
- shell脚本之循环语句与函数
shell脚本之循环语句与函数 echo的用法: echo -n #表示不换行输出 echo -e #输出转义字符,将转义后的内容输出到屏幕上 转义字符: \n :换行,被输出的字符从"\n ...
- java程序出现oom如何解决?什么场景下会出现oom?
1.什么是oom? OOM,全称"Out Of Memory",翻译成中文就是"内存用完了",来源于java.lang.OutOfMemoryError. 当J ...
- Linux命令安装Mysql
关键步骤: 4.创建用户组和用户 groupadd mysql useradd -r -g mysql mysql 5.修改权限 chown -R mysql:mysql ./ 6.安装数据库 ./s ...
- Oracle 11G 安装详解
oracle官网下载地址:http://www.oracle.com/technetwork/database/enterprise-edition/downloads/index.html 官网下载 ...
- ASP.NET与.NET Framework和C#的关系
你好,是我琉忆. 今天我们讲一讲ASP.NET与.NET Framework和C#的关系. 在开始介绍ASP.NET之前,我们需要先了解以下运行ASP.NET的开发平台.NET框架.如果你之前学过C# ...
- Hadoop分布式集群部署
环境准备 IP HOSTNAME SYSTEM 192.168.131.129 hadoop-master CentOS 7.6 192.168.131.135 hadoop-slave1 CentO ...
- mysq数据库相信介绍大纲!!!!!!
什么是数据库? 数据库(Database)是按照数据结构来安排.存储和办理数据的仓库. 每个数据库都有一个或多个不同的 API 用于创立,访问,办理,搜索和仿制所保存的数据. 我们也能够将数据存储在文 ...