宏只是简单的代码替换,不会像函数一样先将参数计算后,再传递
在使用带参数的宏的时候要注意:
所有的参数都加括号,然后整个表达式在加上一个括号
错误示例:
#include <stdio.h>
#define RECTANGLE_AREA1(a, b) a * b
#define RECTANGLE_AREA2(a, b) (a * b)
#define RECTANGLE_AREA3(a, b) (a) * (b)
int main(void)
{
int a = 1;
int b = 2;
int c = 3;
int d = 4;
int base = 10;
int ret = 0;
/* 本意是:base/(a*b),实际宏展开为:base/a*b,造成错误 */
ret = base / RECTANGLE_AREA1(a, b);
/* 本意是:base/((a+b)*(c+d)),实际宏展开为:base/(a+b*c+d),造成错误 */
ret = base / RECTANGLE_AREA2(a + b, c + d);
/* 本意是:base/((a+b)*(c+d)),实际宏展开为:base/(a+b)*(c+d),造成错误 */
ret = base / RECTANGLE_AREA3(a + b, c + d);
return 0;
}
综上,最保险的方法涉及到的参数全部加括号,然后整个表达式再加上一个括号
#define RECTANGLE_AREA(a, b) ((a) * (b))
对于普通的宏(不带参数),也要注意:
如果是正数,则不用加括号
如果是负数,需要将整个负数加括号
如果是表达式,需要整个表达式要加括号
下面的代码都是limits.h头文件中的相关的宏定义
/* Minimum and maximum values a `signed short int' can hold. */
# define SHRT_MIN (-32768)
# define SHRT_MAX 32767
可以看到负数要加括号,正数不需要加括号
我暂时没有想到怎么构造错误的示例,但是负数如果不加括号肯定是有问题的
因为数组下标[],圆括号(),成员选择(对象).,成员选择(指针)->的优先级都比负号要高
/* Minimum and maximum values a `signed int' can hold. */
# define INT_MIN (-INT_MAX - 1)
# define INT_MAX 2147483647
可以看到,如果是一个表达式,那么整个表达式也要加括号
这个错误示例就比较好构造了
#include <stdio.h>
#define INT_MIN -INT_MAX - 1
#define INT_MAX 2147483647
int main()
{
double a = 0.5;
double b = 0;
/* 本意是:(-INT_MAX-1)*a,实际宏展开为:-INT_MAX-1*a,造成错误 */
b = INT_MIN * a;
return 0;
}