宏定义2
1、带参宏和带参函数的区别
(1)宏定义是在预处理期间处理的,而函数是在编译期间处理的。这个区别带来的实质差异是:宏定义最终是在调用宏的
地方把宏体原地展开,而函数是在调用函数处跳转到函数中去执行,执行完后再跳转回来。
注:宏定义和函数最大差别就是:宏定义是原地展开,因此没有调用开销;而函数是跳转执行再返回,因此函数有比较大
的调用开销。所以宏定义和函数相比,优势就是没有调用开销,没有传参开销,所以当函数体很短(尤其是只有一句话)
可以用宏定义来替代,这样效率高。
(2)带参宏和带参函数的一个重要差别就是:宏定义不会检查参数的类型,返回值也不会附带类型;而函数有明确的参数类型
和返回值类型。当我们调用函数时编译器会帮我们做参数的静态类型检查,如果编译器发现我们实际传参和参数声明不一样时就会
报错或者报警告。
注:用函数的时候程序员不太用操心类型不匹配,因为编译器会检查,如果不匹配编译器会有提示;用宏的时候程序员必须很注意
实际传参和宏所希望的参数类型一致,否则可能编译不报错,但是运行有误。
总结:宏和函数各有千秋,各有优劣。总的来说,如果代码比较多用函数适合而且不影响效率;但是对于那些只有一两句话的函数开销
就太大了,适合用带参宏。但是用带参宏又有缺点:不检查参数类型。
2、内联函数和inline关键字
(1)内联函数通过在函数定义前加incline关键字实现。
(2)内联函数本质上是函数,所以有函数的优点(内联函数是编译器负责处理的,编译器帮助我们做参数的静态类型检查);但是他同时
也有带参宏的优点(不用调用开销,而是原地展开)。所以几乎可以这样认为:内联函数就是带了参数静态类型检查的宏。
(3)当我们的函数内函数体很短时(譬如只有一两句话)的时候,我们又希望利用编译器的参数类型检查来排错,还希望没有
调用开销时,最适合使用内联函数。
3、宏定义来实现条件编译(#define #undef #ifdef)
(1)程序中有DEBUG版本和RELEASE版本,区别就是编译时有无定义DEBUG宏。
1 #include <stdio.h> 2 3 #define DEBUG 4 5 //#undef DEBUG //注销一个宏,如果前边有定义这个宏,则注销 6 #ifdef DEBUG 7 #define debug(x) printf(x) 8 #else 9 #define debug(x) 10 #endif 11 12 int main(void) 13 { 14 debug("this is a debug info. "); 15 return 0; 16 }