宏参数(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。

 

posted @ 2022-03-19 16:16  chaoguo1234  阅读(1108)  评论(0编辑  收藏  举报