C ++模板类对于不同的编译器表现不同

时间:2013-10-11 12:57:19

标签: c++ qt templates mingw32 visual-studio-2010

我正在使用C ++中的模板。使用MSVC编译器和使用Mingw gcc编译器时,使用模板和友元类是否有任何区别。我的代码在使用MSVC编译时成功编译并提供所需的输出,但在使用gcc编译时会出错。以下是我的代码,

///////////Record.h/////////////////////
#include "Base.h"

class Derived1;
class Derived2;
template <class TYPE_LIST> class List;

class FRecord
{
public:
    FRecord();
    virtual ~FRecord();

    friend class Base;
#if _MSC_VER <= 1200
    friend class List<Derived1>;
    friend class List<Derived2>;
#else
    template <class TYPE_LIST> friend class List;
#endif
};

///////////////////////////////////////////////////////////////

///////////////////Base.h/////////////////////////////////

class Base
{
public:
    Base(const HEADER *hc, const FRecord *fr);
    virtual ~Base();    
    __inline bool IsNonValid() const;

protected:
    quint32 Size;
};

/////////////////////////////////////
// Data
/////////////////////////////////////
template <class TYPE_LIST>
class Data : public TYPE_LIST
{
public:
    Data(const HEADER *hc, const FRecord *fr) : TYPE_LIST(hc, fr)
    {
        QString val = IsNonValid() ? "Non" : "";
        LOG0("Data ("<<  val << " Valid)");
    }

    virtual ~Data()
    {
        LOG0("Data deleted");
    }
};  // Data

///////////////////////////////////////////////////////////////////////////////////////

编译时,上面的代码用MSVC给出了所需的输出,但是当用Mingw GCC编译器编译时,它会产生以下错误,

Base.h:1154: error: there are no arguments to 'IsNonValid' that depend on a template parameter, so a declaration of 'IsNonValid' must be available
Base.h:1553: error: 'Size' was not declared in this scope

这个问题的可能解决方案是什么? 提前谢谢。

2 个答案:

答案 0 :(得分:10)

MSVC未正确实现两阶段名称查找。 GCC报告此错误是正确的。

原因是模板中使用的名称不依赖于模板的参数(在VC的情况下)在定义模板时查找,而不是在实例化时查找。

在你的情况下,编译器无法告诉IsNonValid将来自基类,所以它理所当然地抱怨它不知道它。有两种可能的解决方案:

  1. 限定对IsNonValid的访问权限,以便编译器清楚它(可能)取决于模板参数:

    QString val = this->IsNonValid() ? "Non" : "";
    
    // or
    
    QString val = TYPE_LIST::IsNonValid() ? "Non" : "";
    
  2. 将继承的名称引入派生类的范围:

    template <class TYPE_LIST>
    class Data : public TYPE_LIST
    {
    public:
      using TYPE_LIST::IsNonValid;
      // the rest as you had it originally
    
  3. 其中任何一个都会使名称依赖,从而将其查找推迟到实施时,实际知道TYPE_LIST的值。

答案 1 :(得分:7)

gcc是对的。您需要添加this->以延迟查找,直到实例化时间。

this->IsNonValid();

MSVC不符合要求,它会延迟所有查找,直到实例化时间,因为它没有正确实现两阶段名称查找。