如果在宏定义中存在较为复杂的语句,直接调用多个语句可能会出现问题:
#include
#define add(a, b) int sum=a+b; printf("sum=%d\n", sum);
int main(){
if(0)
add(1,2);
return 0;
}
这个程序会报编译错误:
error: 'sum' undeclared (first use in this function)
这是由于编译前,通过预处理后,程序的代码被处理成:
#include
#define add(a, b) int sum=a+b; printf("sum=%d\n", sum);
int main(){
if(0)
int sum=1+2; printf("sum=%d\n", sum);
return 0;
}
也就是实际的逻辑是:
#include
#define add(a, b) int sum=a+b; printf("sum=%d\n", sum);
int main(){
if(0)
int sum=1+2;
printf("sum=%d\n", sum);
return 0;
}
由于sum的定义是在if语句块的,因此会报错无法找到变量sum。
可以通过增加大括号解决这个问题:
#include
#define add(a, b) {int sum=a+b; printf("sum=%d\n", sum);}
int main(){
if(0)
add(1,2);
return 0;
}
虽然可以正常的编译运行,但是如果增加了else,又会报错:
#include
#define add(a, b) {int sum=a+b; printf("sum=%d\n", sum);}
int main(){
if(0)
add(1,2);
else
add(3,4);
return 0;
}
error: 'else' without a previous 'if'
这是由于经过预处理后,代码变成了:
#include
#define add(a, b) {int sum=a+b; printf("sum=%d\n", sum);}
int main(){
if(0)
{int sum=1+2; printf("sum=%d\n", sum);};
else
{int sum=3+4; printf("sum=%d\n", sum);};
return 0;
}
if语句块的分号已经将if语句终结,因此else会报错,没有if。
可以通过为每个语句块增加大括号解决问题:
#include
#define add(a, b) {int sum=a+b; printf("sum=%d\n", sum);}
int main(){
if(0)
{
add(1,2);
}
else
{
add(3,4);
}
return 0;
}
虽然问题解决了,但对于使用上来说稍有些麻烦。
可以通过do{...} while(0)方式来更好的解决这个问题:
#include
#define add(a, b) do{int sum=a+b; printf("sum=%d\n", sum);}while(0)
int main(){
if(0)
add(1,2);
else
add(3,4);
return 0;
}
do{...} while(0)方式可以不受大括号、分号的影响,不论在什么情况下使用,宏都可以正常的运行。