C中#define和inline的区别

来源:互联网 发布:淘宝仅退款 编辑:程序博客网 时间:2024/04/20 04:25

原文地址:http://blog.csdn.net/nevasun/article/details/6901641

 C中宏定义和内联函数的区别是很容易让人忽视的一个地方,也是面试的时候经常被提起的一个问题。前几天写代码的时候由于对宏定义一个细节的忽视,导致程序运行总是出错。现在把宏定义可能出现的问题总结一下。
        出错的代码结构如下:

[cpp] view plaincopy
  1. if(RLC_DC_BIT_MSK == data_or_control)  
  2.     LOG_INFO(...);  
  3. else  
  4. {  
  5.     switch(ucPduType)  
  6.     {  
  7.     case RLC_PDU_TYPE_STATUS:  
  8.         LOG_INFO(...);  
  9.         break;  
  10.     case RLC_PDU_TYPE_RESET:  
  11.     case RLC_PDU_TYPE_RESET_ACK:  
  12.         LOG_INFO(...);  
  13.         break;  
  14.     default:  
  15.         LOG_INFO(...);  
  16.         break;  
  17.     }  
  18. }  

        当第一个if条件判断为假时,程序并不会执行else分支。这是为什么呢?
        查看LOG_INFO()的宏定义:

[cpp] view plaincopy
  1. #define LOG_INFO(...) \  
  2.     if(GetLogOutSwitch(SP_LOG_INFO_LEVEL, sessionId)) \  
  3.     if (type == ASN1) \  
  4.     SendLog(...); \  
  5.     else \  
  6.     SendLog(...)  

        根据C语言的编译规则,else分支和LOG_INFO()中的if进行了匹配,导致程序并没有进else分支执行。
        define成为“宏”,在C语言编程中非常重要,它在程序编译时只是在预处理的过程中实施简单的替换操作而已,但是在替换过程中可能出现各种不安全性问题,不进行参数有效性检查。
        内联函数和普通函数相比可以加快程序的运行速度,但它是以增加程序存储空间为代价的,由于不需要中断调用,在编译内联函数的时候内联函数可以直接被嵌入目标代码中。
        对于短小的代码,inline可以带来一定效率的提升,且与C时代的define(宏)相比,它更安全可靠。宏和内联函数的主要区别如下:
        1. 宏是代码处不加任何验证的简单替代,而内联函数是将代码直接插入调用处,而减少了普通函数调用时的资源消耗。
        2. 宏不是函数,只是在编译前预处理阶段将程序中有关字符串替换成宏体。
        3. inline是函数,但在编译中不单独产生代码,而是将有关代码嵌入到调用处。
        总结如下:
        对于一般常量,最好用const和enum替换#define;
        对于类似函数的宏,最好改用inline函数替换#define。
        两个例子:
1.

[cpp] view plaincopy
  1. #define MAX(a, b) ((a) > (b) ? (a):(b)) /* 得到两个数中的最大值 */  
  2. int a = 5, b = 0;  
  3. MAX(++a, b); /* a会被递增两次,最终返回结果是7 */  

2.

[cpp] view plaincopy
  1. #define ABC(x)  (x*x)  
  2. __inline int abc(const int x)   {   return (x*x);   }  
  3.   
  4. printf("%d\n", ABC(1+1));   /* 3 */  
  5. printf("%d\n", abc(1+1));   /* 4 */  

0 0
原创粉丝点击