C中宏定义和内联函数的区别是很容易让人忽视的一个地方,也是面试的时候经常被提起的一个问题。前几天写代码的时候由于对宏定义一个细节的忽视,导致程序运行总是出错。现在把宏定义可能出现的问题总结一下。
根据C语言的编译规则,else分支和LOG_INFO()中的if进行了匹配,导致程序并没有进else分支执行。
define成为“宏”,在C语言编程中非常重要,它在程序编译时只是在预处理的过程中实施简单的替换操作而已,但是在替换过程中可能出现各种不安全性问题,不进行参数有效性检查。
内联函数和普通函数相比可以加快程序的运行速度,但它是以增加程序存储空间为代价的,由于不需要中断调用,在编译内联函数的时候内联函数可以直接被嵌入目标代码中。
对于短小的代码,inline可以带来一定效率的提升,且与C时代的define(宏)相比,它更安全可靠。宏和内联函数的主要区别如下:
1. 宏是代码处不加任何验证的简单替代,而内联函数是将代码直接插入调用处,而减少了普通函数调用时的资源消耗。
2. 宏不是函数,只是在编译前预处理阶段将程序中有关字符串替换成宏体。
3. inline是函数,但在编译中不单独产生代码,而是将有关代码嵌入到调用处。
总结如下:
对于一般常量,最好用const和enum替换#define;
对于类似函数的宏,最好改用inline函数替换#define。
两个例子:
转载自 http://blog.csdn.net/nevasun/article/details/6901641
出错的代码结构如下:
if(RLC_DC_BIT_MSK == data_or_control)
LOG_INFO(...);
else
{
switch(ucPduType)
{
case RLC_PDU_TYPE_STATUS:
LOG_INFO(...);
break;
case RLC_PDU_TYPE_RESET:
case RLC_PDU_TYPE_RESET_ACK:
LOG_INFO(...);
break;
default:
LOG_INFO(...);
break;
}
}
当第一个if条件判断为假时,程序并不会执行else分支。这是为什么呢?
查看LOG_INFO()的宏定义
#define LOG_INFO(...) \
if(GetLogOutSwitch(SP_LOG_INFO_LEVEL, sessionId)) \
if (type == ASN1) \
SendLog(...); \
else \
SendLog(...)
根据C语言的编译规则,else分支和LOG_INFO()中的if进行了匹配,导致程序并没有进else分支执行。
define成为“宏”,在C语言编程中非常重要,它在程序编译时只是在预处理的过程中实施简单的替换操作而已,但是在替换过程中可能出现各种不安全性问题,不进行参数有效性检查。
内联函数和普通函数相比可以加快程序的运行速度,但它是以增加程序存储空间为代价的,由于不需要中断调用,在编译内联函数的时候内联函数可以直接被嵌入目标代码中。
对于短小的代码,inline可以带来一定效率的提升,且与C时代的define(宏)相比,它更安全可靠。宏和内联函数的主要区别如下:
1. 宏是代码处不加任何验证的简单替代,而内联函数是将代码直接插入调用处,而减少了普通函数调用时的资源消耗。
2. 宏不是函数,只是在编译前预处理阶段将程序中有关字符串替换成宏体。
3. inline是函数,但在编译中不单独产生代码,而是将有关代码嵌入到调用处。
总结如下:
对于一般常量,最好用const和enum替换#define;
对于类似函数的宏,最好改用inline函数替换#define。
两个例子:
#define MAX(a, b) ((a) > (b) ? (a):(b)) /* 得到两个数中的最大值 */
int a = 5, b = 0;
MAX(++a, b); /* a会被递增两次,最终返回结果是7 */
#define ABC(x) (x*x)
__inline int abc(const int x) { return (x*x); }
printf("%d\n", ABC(1+1)); /* 3 */
printf("%d\n", abc(1+1)); /* 4 */
转载自 http://blog.csdn.net/nevasun/article/details/6901641