1. 函数宏介绍
函数宏,即包含多条语句的宏定义,其通常为某一被频繁调用的功能的语句封装,且不想通过函数方式封装来降低额外的弹栈压栈开销。
函数宏本质上为宏,可以直接进行定义,例如:
#define INT_SWAP(a,b) \
int tmp = a; \
a = b; \
b = tmp
但上述的宏具有一个明显的缺点:当遇到 if
、while
等语句且不使用花括号仅调用宏时,实际作用范围在宏的第一个分号后便结束。即 a = b
和 b = tmp
均不受控制语句所作用。
因此,在工程中,一般使用三种方式来对函数宏进行封装,分别为 {}
、do{...}while(0)
和 ({})
。下文将一一对三种方式进行分析,比较各自的优劣点。
2. {}
方式
INT_SWAP
宏使用 {}
封装后形态如下:
#define INT_SWAP(a,b)\
{ \
int tmp = a; \
a = b; \
b = tmp; \
}
此时,直接调用与在无花括号的控制语句(如 if
、while
)中调用均能正常运行,例如:
#define INT_SWAP(a,b) \
{ \
int tmp = a; \
a = b; \
b = tmp; \
}
int main()
{
int var_a = 1;
int var_b = 2;
INT_SWAP(var_a, var_b);
printf("var_a = %d, var_b = %d\n", var_a, var_b); // var_a = 2, var_b = 1
if (1)
INT_SWAP(var_a, var_b);
printf("var_a = %d, var_b = %d\n", var_a, var_b); // var_a = 1, var_b = 2
}
但当无花括号的 if
语句存在其他分支(else if
、else
等)如:
if