C语言:秀操作 | 函数宏的三种封装方式
1. 函数宏介绍
函数宏本质上为宏,可以直接进行定义,例如:
  
    
    
  
   
     
     
       #define INT_SWAP(a,b) \
    int tmp = a;          \
    a = b;                \
    b = tmp;              \
  
    
    
    
 因此,在工程中,一般使用三种方式来对函数宏进行封装,分别为 {}、do{…}while(0) 和 ({})。下文将一一对三种方式进行分析,比较各自的优劣点。
2. {} 方式
  
    
    
  
   
     
     
   #define INT_SWAP(a,b)\
{                   \
    int tmp = a;    \
    a = b;          \
    b = tmp;        \
}
  
    
    
    
 
  
    
    
  
   
     
     
   #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 (1)
   INT_SWAP(var_a, var_b);
else
 printf("hello world!\n");
  
    
    
    
 
  
    
    
  
   
     
     
   ...
/mnt/hgfs/share/pr_c/src/main.c: In function ‘main’:
/mnt/hgfs/share/pr_c/src/main.c:18:2: error: ‘else’ without a previous ‘if’else
  
    
    
    
 因此,解决方法有两种,分别为不使用 ;(port.1)或规定必须使用带花括号的 if(port.2),例如:
  
    
    
  
   
     
     
   /* port.1 */
if (1)
   INT_SWAP(var_a, var_b)
else
{
    printf("hello world!\n");
}
/* port.2 */
if (1)
{
   INT_SWAP(var_a, var_b);
}
else
{
    printf("hello world!\n");
}
  
    
    
    
 优缺点总结:
优点:简单粗暴。
缺点:不能在无花括号且有分支的 if 语句中直接调用;能够不带 ; 直接调用。
3. do{…}while(0) 方式
  
    
    
  
   
     
     
   #define INT_SWAP(a,b)   \
do{                     \
    int tmp = a;        \
    a = b;              \
    b = tmp;            \
}while(0)
  
    
    
    
 由于 do{…}while(0) 实际为 while 循环,因此可以使用关键字 break 提前结束循环。利用该特性,可以为函数宏添加参数检测。例如:
  
    
    
  
   
     
     
   #define INT_SWAP(a,b)  \
do{                 \
 if (a < 0 || b < 0) \
  break;   \
    int tmp = a;     \
    a = b;           \
    b = tmp;         \
}while(0)
  
    
    
    
 因此,do{…}while(0) 方式的函数宏可以在无花括号且有分支的 if 语句中直接调用。例如:
  
    
    
  
   
     
     
   #define INT_SWAP(a,b)  \
do{                 \
 if (a < 0 || b < 0) \
  break;   \
    int tmp = a;     \
    a = b;           \
    b = tmp;         \
}while(0)
int main()
{
 int var_a = 1;
 int var_b = 2;
 if (1)
    INT_SWAP(var_a, var_b);
 else
  printf("hello world!\n"); 
 printf("var_a = %d, var_b = %d\n", var_a, var_b); // var_a = 2, var_b = 1
 return 0;
}
  
    
    
    
 
  
    
    
  
   
     
     
   if (1)
   INT_SWAP(var_a, var_b)
else
{
 printf("hello world!\n"); 
}
  
    
    
    
 优点:支持在无花括号且有分支的 if 语句中直接调用;支持提前退出函数宏;强制调用时必须使用 ;。
缺点:无返回值,不能作为表达式的右值使用。
4. ({}) 方式
INT_SWAP 宏使用 ({}) 封装后形态如下:
  
    
    
  
   
     
     
   #define INT_SWAP(a,b)   \
({                      \
    int tmp = a;        \
    a = b;              \
    b = tmp;            \
})
  
    
    
    
 
  
    
    
  
   
     
     
   #define INT_SWAP(a,b)  \
({                 \
 int tmp = a;    \
 a = b;          \
 b = tmp;        \
})
int main()
{
 int var_a = 1;
 int var_b = 2;
 if (1)
    INT_SWAP(var_a, var_b);
 else
  printf("hello world!\n");
 printf("var_a = %d, var_b = %d\n", var_a, var_b); // var_a = 2, var_b = 1
 return 0;
}
  
    
    
    
 答案是 C 语言规定 ({}) 中的最后一条语句的结果为该双括号体的返回值。例如:
  
    
    
  
   
     
     
   int main()
{
 int a = ({
  10;
  1000;
 });
 printf("a = %d\n", a);      // a = 1000
}
  
    
    
    
 
  
    
    
  
   
     
     
   #define INT_SWAP(a,b)  \
({                 \
 int ret = 0;  \
 if (a < 0 || b < 0) \
 {     \
  ret = -1;  \
 }     \
 else    \
 {     \
  int tmp = a;    \
  a = b;          \
  b = tmp;        \
 }     \
 ret;    \
})
int main()
{
 int var_a = 1;
 int var_b = 2;
 if (INT_SWAP(var_a, var_b) != -1)
  printf("swap success !!\n");     // swap success !!
 else
  printf("swap fail !!\n"); 
 printf("var_a = %d, var_b = %d\n", var_a, var_b); // var_a = 2, var_b = 1
 return 0;
}
  
    
    
    
 优缺点总结:
优点:支持在无花括号且有分支的 if 语句中直接调用;有返回值,支持作为表达式的右值。
缺点:不支持提前退出函数宏;非 C 的原生语法,编译器可能不支持。
