7

我正在使用 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

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

4

2 回答 2

11

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
    

这些中的任何一个都会使名称依赖并因此将其查找推迟到实例化,此时TYPE_LIST实际知道 的值。

于 2013-10-11T13:04:18.253 回答
7

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

this->IsNonValid();

MSVC 的不一致之处在于它将所有查找延迟到实例化时间,因为它没有正确实现两阶段名称查找。

于 2013-10-11T13:02:28.727 回答