本文主要是介绍宏的副作用和do{}while(0)的说明,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
本文如下:
问题:写一个“标准”宏MIN,这个宏输入两个参数并返回较小的一个,另外,当你写下面的代码时会发生什么事?
least=MIN(*p++,b);
解答:
#define MIN(A,B) ((A)<=(B)?(A):(B)) //注意加括号的方法
当使用MIN(*p++,b)时会产生宏的副作用;
剖析:
这个面试题主要考察面试者对宏定义的使用,宏定义可以实现类似于函数的功能,但是它终归不是函数,而宏定义中括弧中的"参数"也不是真的参数,在宏展开时对"参数"进行的是一对一的替换;
程序员对宏定义的使用要非常小心,特别要注意两个问题:
(1)、谨慎地将宏定义中的"参数"和整个宏用括号扩起来;
所以,严格地讲,下述解答都是错误的:
#define MIN(A,B) (A)<=(B)?(A):(B)
#define MIN(A,B) (A<=B?A:B)
(2)、防止宏的副作用:
#define MIN(A,B) ((A)<=(B)?(A):(B))对于MIN(*p++,b)的作用结果为((*p++)<=(b)?(*p++):(b)) *p会做两次自加操作;
除此之外,另一个错误的解答是#define MIN(A,B) ((A)<=(B)?(A):(B));这个解答在宏定义的后面加了一个逗号,显示编写者对宏的概念模糊不清,也是错误的;
#include <stdio.h>
#include<stdlib.h>
#define min_i(x, y) ((x) < (y) ? (x) : (y))//(1)
#define min_t(type, x, y) do{ type __x = x; \
type __y = y;\
__x < __y ? __x : __y;}while(0);
#define min(x, y) { const int _x = (x); const int _y = (y); (void) (&_x == &_y);\
printf("%d\n",_x < _y ? _x : _y); };
int main()
{
int a = 10; int b = 20;
printf("min_i(a++, b++) = %d\n", min_i(a++, b++)); // 11
printf("a = %d\n",a); // 12
printf("b = %d\n",b); // 21
int c = 10; char d = 'a';
min_t(int, c++, d++); // 10
printf("a = %d\n", a); // 11
printf("b = %d\n", b); // 21
a = 10; b = 20;
min(a++, b++); // 10
printf("a = %d\n", a); // 11
printf("b = %d\n", b); // 21
system("PAUSE");
return 0;
}
(1)、这种定义计算x和y分别两次(其实是x和y中的小者被计算两次),当参数有副作用时,将产生不正确的结果;
(2)、后面两种定义只计算参数一次,避免了可能的错误:语句表达式通常用于宏定义;
do{}while(0)的作用:http://www.cnblogs.com/flying_bat/archive/2008/01/18/1044693.html
来源:
http://wenku.baidu.com/link?url=-vpjW5-L5AhYC60PBUteUgoVTtZsLgnGDVeh9pAVLu4PQK4Y4rucfzyVzwDmDHRTzXL3uRqz3RoK-WoYgCRmjUEmPkKkNkk-EjKGaiQ3hTy(自己做了一些修改)
这篇关于宏的副作用和do{}while(0)的说明的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!