15 September 2014
如果让你写一个求最小值的宏定义时,大部分人可能觉得很简单,心想只要注意加上括号不久可以么,信手拈来,刷刷写下:
#define min(x, y) ((x) < (y) ? (x) : (y))
确实在大部分情况下,这个宏定义已经足够使用,我们考虑这么一种情况:
int a = 3;
int b = 4;
printf("%d\n", min(a++,b++));
printf("a = %d,b = %d", a, b);
代码中将将a++和b++作为参数传入,本是先比较a和b,然后将a和b加1,运行这段代码会发现首先输出最小值4,然后a等于5,b等于5。问题就出在++运算符上。min(x++,y++)被替换为((a++) < (b++) ? (a++) : (b++)),首先执行(a++) < (b++)语句,比较a和b,比较完成之后两者都加1,然后输出时因为输出的是a,此时返回的值是a++,于是先返回4之后,a又加1。
那我们如何避免这种情况呢?我们可以用两个值来保存两个参数,然后使用两个参数进行计算,那这两个参数的类型我们如何得知呢?可以通过typeof关键字,这个关键字是GNU 扩展的C语言关键字,VC不支持。我们看下代码: