对于以下代码,我有一个奇怪的编译警告,使用Visual C ++ 2010:
#include <iostream>
class test
{
public:
template<class obj>
class inner
{
private:
// Line 11:
template<int index, bool unused = true> struct AttributeName;
private:
template<bool b>
struct AttributeName<0,b>
{
static inline const char* get()
{
return "prop";
}
};
public:
typedef AttributeName<0> propname;
};
typedef inner<test> description;
};
int main()
{
test t;
std::cout << test::description::propname::get(); // Line 32
return 0;
}
警告:
file.cpp(11) : warning C4348: 'test::inner<obj>::AttributeName' : redefinition of default parameter : parameter 2 (with [ obj=test ])
file.cpp(11) : see declaration of 'test::inner<obj>::AttributeName' (with [ obj=test ])
file.cpp(32) : see reference to class template instantiation 'test::inner<obj>' being compiled (with [ obj=test ])
我不明白的是AttributeName
“重新定义”与定义在同一行...听起来像一个错误
我注意到将inner
设为非模板类会删除警告。但是,这不是一个选项,因为真正的代码比这个测试用例更复杂,需要模板化。
此外,如果警告被视为错误,则此代码将无法编译...
It compiles without warnings on GCC
为什么msvc会输出这样的警告并且有解决方法吗?
修改
以下修改:
template<int index, bool unused = true> struct AttributeName {};
似乎抹去了警告。
答案 0 :(得分:8)
这主要是猜测,因为这似乎解决了(?)它:
template<int index, bool unused = true> struct AttributeName;
template<int index, bool unused> struct AttributeName
{
};
我的猜测是前向声明被视为声明和“定义”,因为它看不到任何其他内容,因此它抱怨默认为“重新定义”,即使它是同一行。可能是无心的,虽然2012年表现相同。