C++ 标准库函数与宏定义的名字冲突

转至  http://blog.csdn.net/rnamatrix/article/details/5765462

今天在使用limits中的numeric_limits类模板函数max和min时,编译出现了一个错误:

  
  
[c-sharp] view plain copy print ?
  1. error C2589: '(' : illegal token on right side of '::'
百思不得其解,后来突然想到max和min很有可能已经被定义成了宏,于是查看包进来的头文件,发现蹊跷肯定在<windows.h>中:
windows.h包括了windef.h头文件,在windef.h中定义有宏:
  
  
[c-sharp] view plain copy print ?
  1. #ifndef NOMINMAX
  2. #ifndef max
  3. #define max(a,b) (((a) > (b)) ? (a) : (b))
  4. #endif
  5. #ifndef min
  6. #define min(a,b) (((a) < (b)) ? (a) : (b))
  7. #endif
  8. #endif /* NOMINMAX */
所以这里的max和min的宏定义就与标准模板库中的numeric_limits<*>::max/min的定义发生了冲突,加入#define NOMINMAX问题就解决了。
(注意宏的作用域都是全局(scope-less evil)的)
进一步去google了一下关于类似的max/min宏的定义还发生在哪些常用的头文件中。
这里指出,max/min的宏定义还出现在了stdlib.h和minmax.h头文件中。
在stdlib.h中:
  
  
[c-sharp] view plain copy print ?
  1. #if !__STDC__
  2. #ifndef _POSIX_
  3. /* Non-ANSI names for compatibility */
  4. #ifndef __cplusplus
  5. #define max(a,b) (((a) > (b)) ? (a) : (b))
  6. #define min(a,b) (((a) < (b)) ? (a) : (b))
  7. #endif
  8. ...
一般ANSI C中使用__max和__min宏,max和min只是在非标准C中才会定义,所以一般使用标准C时,包括了stdlib.h头文件不会发生和max/min宏的冲突。
而minmax.h头文件一般只是在你想使用max/min宏时才会被包括进来。
并且,在网上还发现了此问题的另外一个解决方式:
  
  
[c-sharp] view plain copy print ?
  1. (std::min)(x, y);
  2. (std::max)(x, y);
  3. (std::numeric_limits<T>::min)();
  4. (std::numeric_limits<T>::max)();
这样把函数名用括号括起来了,max/min不再被当作带参数的宏去替换了,因此能够避免冲突。
当用户自己定义的类型的成员函数与全局的宏定义发生冲突时,都可以采用这种解决方法。
  
  
[c-sharp] view plain copy print ?
  1. template <typename T, int Size>
  2. struct Series
  3. {
  4. T min() { return *(std::min_element(s, s + Size); }
  5. T& operator[](int index) { return s[index]; }
  6. private:
  7. T s[Size];
  8. };
  9. Series<int, 3> s;
  10. s[0] = 2;
  11. s[1] = 3;
  12. s[2] = 1;
  13. int m = (s.min)(); // long way, but here is the trick
只是,这种方式可能会面临一些负面影响:
这种方式下不能使用ADL(Argument Depended Name Lookup)了,所以成员函数前必须加上类名或者名字空间域名,也就是必须是fully qualified。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值