C中#define和inline的区别

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

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。
        两个例子:
1.

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

2.

#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 */

你可能感兴趣的:(编程,c,面试,存储,语言)