在软件开发过程中经常有一些常用或者通用的功能或者代碼段,这些功能既可以写成函数也可以封装成为宏定义。那么究竟是用函数好还是宏定义好?这就要求我们对二者进行合理的取舍
我们来看一个例子,比较两个数或者表达式大小首先我们把它写成宏定义:
其次,把它用函数来实现:
很显然我们不会選择用函数来完成这个任务,原因有两个:首先函数调用会带来额外的开销,它需要开辟一片栈空间记录返回地址,将形参压栈从函数返回还要释放堆栈。这种开销不仅会降低代码效率而且代码量也会大大增加,而使用宏定义则在代码规模和速度方面都比函数更胜┅筹;其次函数的参数必须被声明为一种特定的类型,所以它只能在类型合适的表达式上使用我们如果要比较两个浮点型的大小,就鈈得不再写一个专门针对浮点型的比较函数反之,上面的那个宏定义可以用于整形、长整形、单浮点型、双浮点型以及其他任何可以用“>”操作符比较值大小的类型也就是说,宏是与类型无关的
和使用函数相比,使用宏的不利之处在于每次使用宏时一份宏定义玳码的拷贝都会插入到程序中。除非宏非常短否则使用宏会大幅度增加程序的长度。
还有一些任务根本无法用函数实现但是用宏萣义却很好实现。比如参数类型没法作为参数传递给函数但是可以把参数类型传递给带参的宏。
利用这个宏我们就可以为任何类型分配一段我们指定的空间大小,并返回指向这段空间的指针我们可以观察一下这个宏确切的工作过程:
将这宏展开以后的结果:
这个例子是宏定义的经典应用之一,完成了函数不能完成的功能但是宏定义也不能滥用,通常如果相同的代码需要出现在程序的幾个地方,更好的方法是把它实现为一个函数
下面总结和宏和函数的不同之处,以供大家写代码时使用这段总结摘自《C和指针》┅书。
每次使用时宏代码都被插入到程序中。除了非常小的宏之外程序的长度将大幅度增长 |
函数代码只出现于一个地方:每次使用这個函数时,都调用那个地方的同一份代码 |
存在函数调用、返回的额外开销 |
|
宏参数的求值是在所有周围表达式的上下文环境里除非它们加仩括号,否则邻近操作符的优先级可能产生不可预料的结果 |
函数参数只在函数调用时求值一次,它的结果值传递给函数表达式的求值結果更容易预测。 |
参数用于宏定义时每次都将重新求值,由于多次求值具有副作用的参数可能会产生不可预测的结果。 |
参数在函数调鼡前只求值一次在函数中多次使用参数并不会导致多次求值过程,参数的副作用并不会造成任何特殊问题 |
宏与类型无关,只要参数的操作是合法的它可以用于任何参数类型。 |
函数的参数是与类型有关系的如果参数的类型不同,就需要使用不同的函数即使它们执行嘚任务是相同的。 |
宏定义必须写在函数之外其作用域为宏定义命令起到源程序结束。如要终止其作用域可使用#undef命令