这里我们简单讨论一下这个问题,我是在打算在 Win10 下进行一个深度学习项目编译时遇到的问题,在该项目中 C++ 自定义op 的 .cpp
文件中使用了 <limits>
库中的 std::numeric_limits<T>
模板进行极值的自定义(可以保证类型安全性),在程序中报了这样的错误:
C:\Users\RyuSein\Anaconda3\Lib\site-packages\tensorflow_core\include\tensorflow/core/protobu
f/config.pb.h(179): warning C4003: 类函数宏的调用“min”参数不足
C:\Users\RyuSein\Anaconda3\Lib\site-packages\tensorflow_core\include\tensorflow/core/protobu
f/config.pb.h(179): error C2589: “(”:“::”右边的非法标记
C:\Users\RyuSein\Anaconda3\Lib\site-packages\tensorflow_core\include\tensorflow/core/protobu
f/config.pb.h(179): error C2062: 意外的类型“unknown-type”
C:\Users\RyuSein\Anaconda3\Lib\site-packages\tensorflow_core\include\tensorflow/core/protobu
f/config.pb.h(179): error C3805: “类型”: 意外标记,应输入“}”或者“,”
而事实上,是因为编译器在预处理阶段,事先已经将 min
认为是 window.h
中包含的一个头文件,windef.h
中的宏定义:
#ifndef NOMINMAX
#ifndef max
#define max(a,b) (((a) > (b)) ? (a) : (b))
#endif
#ifndef min
#define min(a,b) (((a) < (b)) ? (a) : (b))
#endif
#endif
而不是该文件中包含的头文件中定义的模板类型,导致了宏定义冲突的问题。
这里给出我采取的解决方法:
(std::min)(x, y);
(std::max)(x, y);
(std::numeric_limits<T>::min)();
(std::numeric_limits<T>::max)();
通过将函数名用括号包括起来可以防止在预处理的时候被当作宏定义替换。
当然,直接在 windef.h
中去取消这个宏定义是一个比较根本的解决方法,而不用为每个出问题的代码文件都做出一样的修改,但是这样存在风险,因为你不知道你的修改会不会让其他用到这个宏定义的代码文件无法正常编译。如果有更好的方法,还请各位多多指教 人人人。