错误 C2899:不能在模板声明之外使用类型名称

5

我正在尝试在MSV2010中进行以下操作

namespace statismo {
template<>
struct RepresenterTraits<itk::Image<itk::Vector<float, 3u>, 3u> > {

    typedef itk::Image<itk::Vector<float, 3u>, 3u> VectorImageType;
    typedef VectorImageType::Pointer DatasetPointerType;
    typedef VectorImageType::Pointer DatasetConstPointerType;
    typedef typename VectorImageType::PointType PointType;
    typedef typename VectorImageType::PixelType ValueType;
};

我遇到了以下错误:
错误 C2899:typename 不能在模板声明之外使用。
希望能提供一个解决方法,谢谢。
1个回答

11
namespace statismo {
template<>
struct RepresenterTraits<itk::Image<itk::Vector<float, 3u>, 3u> > {

    // bla

    typedef typename VectorImageType::PointType PointType;
            ^^^^^^^^

    typedef typename VectorImageType::PixelType ValueType;
            ^^^^^^^^
};

您的typename关键字位于itk::Image<itk::Vector<float, 3u>, 3u>的显式特化内的RepresenterTraits<T>中。然而,这是一个普通的类,不是一个类模板。这意味着VectorImageType不是一个依赖名称,编译器知道PixelType是一个嵌套类型。这就是为什么不允许使用typename的原因。请参见此 Q&A
请注意,在C++11中,这个限制已被取消,并且在非模板上下文中使用typename允许但不是必需的。例如,参见这个例子。
#include <iostream>

template<class T>
struct V 
{ 
    typedef T type; 
};

template<class T>
struct S
{
    // typename required in C++98/C++11
    typedef typename V<T>::type type;    
};

template<>
struct S<int>
{
    // typename not allowed in C++98, allowed in C++11
    // accepted by g++/Clang in C++98 mode as well (not by MSVC2010)
    typedef typename V<int>::type type;    
};

struct R
{
    // typename not allowed in C++98, allowed in C++11
    // accepted by g++ in C++98 mode as well (not by Clang/MSVC2010)
    typedef typename V<int>::type type;    
};

int main()
{
}
实时示例(只需使用g++/clang和std=c++98/std=c++11命令行选项进行操作)。

非常感谢,这使得情况清晰了,尽管这不是我的代码,我也不完全清楚我可以做什么来快速解决问题。 - Tinashe Mutsvangwa
@TinasheMutsvangwa 很高兴能帮到你! - TemplateRex
@TinasheMutsvangwa 快速的解决方法是在这两个最后的typedef中删除typename。虽然这不是你的代码,但有什么阻止你编辑它的吗? - François Moisan
@FrançoisMoisan:我可以编辑代码,但我不确定它会对其他部分产生什么影响。我已经尝试了您的建议,没有编译错误,但我仍然想知道在执行中的区别是什么。 - Tinashe Mutsvangwa

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接