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

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

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

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

你可能感兴趣的:(C++ 标准库函数与宏定义的名字冲突)