#define 宏是在代码中不加任何验证的直接替代,当宏中包含运算符时要在最外层加括号,不然可能会出错
下面使用一个简单的例子来验证一下,代码中定义一个求两个数中最大数的宏:
MAX_wrong在外层没有加括号,MAX_right在外层加了括号
#include<stdlib.h>
#include<stdio.h>
#define MAX_wrong(x,y) x>y?x:y
#define MAX_right(x,y) (x>y?x:y)
int main()
{
int x, y;
scanf("x=%d y=%d", &x, &y);
//下面这行等价于 int a = x>y?x:y*6
int a = MAX_wrong(x,y) * 6;
//下面这行等价于 int a = (x>y?x:y)*6
int b = MAX_right(x,y) * 6;
/*下面这行输出的两个结果是一样的*/
printf("%d %d\n", MAX_wrong(x,y), MAX_right(x,y));
/*当x>y时,a和b值不一样,当x<=y时,a和b的值一样*/
printf("%d %d\n", a, b);
return 0;
}
经过下面两个测试用例,发现不管定义宏时外层加没加括号,宏MAX_wrong和宏MAX_right的值都是一样的,没有问题。
但是要对宏进行运算时出现了问题,下面第一个的测试结果中,a和b的值碰巧一样;但是第二个测试结果中a和b的值不同,b是我们想要的值,a的值和预想中不一样。
分析其原因:
int a = MAX_wrong(x,y) * 6; 这句话实际相当于 int a = x>y?x:y* 6;
x>y?x:y* 6 的意思是如果x>y则返回x,否则返回冒号后面的值y*6
而int b = MAX_right(x,y) * 6; 这句话实际相当于 int a = (x>y?x:y) * 6;
括号的优先级要高于*,所以会先计算x>y?x:y的值,再进行*6,与我们想要的一样。
完毕!