带参数的宏定义的一般形式如下:
#define 宏名(参数表) 宏体
单看“宏名(参数表)”这个部分就类似于一个“简写”的函数声明,之所以说是“简写”,那是因为,函数声明时,不管是函数名,还是参数名,都会带上类型,如:void add(int a , int b)
{函数体};而用宏则是:#define add(a,b) 宏体
还需注意的是带参数的宏定义的参数表外面本来是有一个括号的,他的意义是表示这是这里是一个参数表,仅此而已。
举一个简单的例子:
#define add(a,b) a + b//定义部分
c = add(a,b);//使用部分使用形式非常类似函数,但是他其实只是简单的替换add(a,b)被替换成了a + b。
再来,举一个例子:
#define SQR(x) (x*x)//定义部分
b = 3;
a =SQR(b+2)//使用部分,猜测一下结果是多少?
在这个宏中,参数表中只有一个参数x,这样给人的感觉就是x是一个整体,而且还被一个括号括起来了。
所以一看看去结果就是(3+2)*(3+2)结果是25.
但是,前面已经强调过了参数表外边的括号,仅仅表示这是这里是一个参数表而已。所以真正的替换结果应该是3+2*3+2,所以编译后得到的结果是11.而不是我们期望的25(求平方的结果)。
那么应该如何得到我们期望的结果呢?
#define SQR((x)) (x*x)为了强调x是一个整体我在参数表中将x外在加上一层括号。但是很可以这种语法是错误的参数表里只能是参数逗号和参数,不能有其他的东西。
所以,正确的写法应该是 #define SQR(x) ((x)*(x)) 这样才会被替换成 (3+2)*(3+2)结果是25.
但是,你说我就这么定义#define SQR(x) (x*x) 但是我使用的时候,这么写SQR((b+2)),这样写编译通过,而且得到的结果是25.
总结:带参数的宏定义,虽然形式上简单,但是他其实还是简单的替换,所以在宏体中,能括起来的尽量都括起来。