C语言中带参数的宏
带参数的宏定义有如下的格式:
【#define 指令----带参数的宏】 #define 标识符(x1,x2,……,xn)
其中 x1,x2,……xn是标志符(宏的参数)
注意:在宏的名字和括号之间必修没有空格。
如果有空格,预处理会认为是在定义一个简单的宏,其中(x1,x2,……,xn)是替换列表的一部分
当预处理器遇到一个带参数的宏,会将定义存储起来以便以后使用。在后面的程序中,如果任何地方出现了标识符(y1,y2……,yn)格式的宏调用(其中y1,y2, ……yn是一些列标记),预处理器会使用替换列表替代,并使用yi替换xi
e.g. 假如我们定义了如下的宏:
#define MAX(x,y) ((x)>(y) ? (x) : (y)) #define IS_EVEN(n) ((n)%2==0)
下面的例子是一个更加复杂的宏:
#define TOUPPER(c) ('a'<=(c) && (c)<='z' ? (c)-'a'+'A' : (c))
带参数的宏可以包含空的参数列表,如下所示:
#define getchar() getc(stdin)
空的参数列表不是一定确实需要,但可以使getchar更像一个函数
使用带参数的宏替代实际的函数有两个优点:
- 程序可能会稍微快些。一个函数调用在执行时通常会有些额外开销----存储上下文信息、复制参数的值等。而一个宏的调用则没有这些运行开销
- 宏会更“通用”。与函数的参数不同,宏的参数没有类型。因此,只要预处理后的程序依然合法的,宏可以接受任何类型的参数。 e.g.我们可以使用MAX宏从两个数中选出较大的一个,数的类型可以是:int,long int,float,double等等
- 编译后的代码通常会变大。每一处宏调用都会导致插入宏的替换列表,由此导致程序源代码增加(因此编译后的代码变大)。宏使用得越频繁,这种效果就越明显。当宏调用嵌套时,这个问题会相互叠加从而使程序更加复杂。e.g. n=MAX(i,MAX(j,k));
下面是预处理后的这条语句:
n=((i)>(((j)>(k)?(j):(k)))?(i):(((j)>(k)?(j):(k))));
- 宏参数没有类型检查。
- 无法用一个指针来指向一个宏
- 宏可能会不止一次地计算它的参数。函数对它的参数只会计算一次,而宏可能会计算两次甚至更多。如果参数有副作用,多次计算参数的值可能会产生意外的结果
考虑下面的例子,其中MAX的一个参数有副作用:
n=MAX(i++, j)
下面是这条语句在预处理之后的结果:
n=((i++) >(j)?(i++):(j));
带参数的宏不仅适用于模拟函数的调用,他们特别经常被作为模板,来处理我们经常要重复书写代码段:
e.g.我们可以使用
#define PRINT_INT(x) printf("%d\n",x);
使得PRINT_INT(x)代替每次使用的printf("%d\n",x);
作者:cpoint
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利.