C ++在模板编译器错误中使用模板

时间:2015-08-06 19:40:29

标签: c++ templates

所以我第一次学习如何使用模板。我正在尝试创建简化的矢量和点类。不幸的是,我需要建模一个点可以通过矢量转换,并且由于两个点的减法可以创建矢量。

我遇到的问题是我需要将文件包含在彼此中。当我在我的点模板头中包含我的矢量类时,我得到编译器错误,因为Point类不再被识别为模板。

Vector Class(在Vector3D.h中)

#include "Point3D.h"

template <class T>
class Vector3D
{
public:
    //Members
    T x,y,z;

    ... //more boring functions

    Point3D<T> operator+ (const Point3D<T>& pt)
    {
        Point3D <T> pt2(x + pt.x, y + pt.y, z + pt.z);
        return pt2;
    }
};

Point Class(在Point3D.h中)

#include "Vector3D.h"

template <class K>
class Point3D
{
public:
    //Members
    K x,y,z;

    ...//boring functions of little importance

    // shifts point by a vector
    Point3D<K> operator+ (const Vector3D<K>& other)
    {
        return Point3D<K>(x + other.x, y + other.y, z + other.z);
    }

    //creates a vector from the differnce between two points
    Vector3D<K> operator- (const Point3D<K>& rhs)
    {
        return Vector3D<K>(x-lhs.x,y-lhs.y,z-lhs.z)
    }
};

现在我收到以下编译器错误(英特尔C ++ 15):

1>F:\Dev Repos\Vascular Modeling\Radiation Modeling Projects\CGAL BOOST  INTEL project1\Vector3D.h(197): error : Point3D is not a template
1>      Point3D<T> operator+ (const Point3D<T>& pt)
1>      ^
1>  
1>F:\Dev Repos\Vascular Modeling\Radiation Modeling Projects\CGAL BOOST INTEL project1\Vector3D.h(197): error : Point3D is not a template
1>      Point3D<T> operator+ (const Point3D<T>& pt)

我做错了什么?我假设我试图打破物理定律并创造某种循环。解决方案是实现拆分文件吗?

谢谢,

威尔

1 个答案:

答案 0 :(得分:0)

如评论中所述,您希望使用前向声明。对于模板类,这些看起来像这样:

template <class K> class Point3D; //before defining Vector3D

template <class T> class Vector3D; //before defining Point3D

假设您还正确使用了包含警戒(如果您在这种情况下不使用包含警卫,则标题将无限期地包含在一起,您甚至会得到更奇怪的错误消息),这应该使一切都正确编译。与非模板类不同,您不必将方法的实现移动到其他文件中,因为它们是在实例化模板时生成的,而不是在定义模板时生成的。

但是,作为旁注,通常建议将二进制操作定义为独立函数而不是类成员,因为隐式转换的规则不同且稍微更直观。因此,我建议改为做这样的事情:

//shifting points by vectors
template <class T>
Point3D<T> operator+ (const Vector3D<T>& vec, const Point3D<T>& pt)
{
    return {vec.x + pt.x, vec.y + pt.y, vec.z + pt.z};
}
template <class T>
Point3D<T> operator+ (const Point3D<T>& pt, const Vector3D<T>& vec)
{
    return vec + pt;
}

//getting a vector from the difference of points
template <class T>
Vector3D<T> operator- (const Point3D<T>& a, const Point3D<T>& b)
{
    return {a.x - b.x, a.y - b.y, a.z - b.z};
}

这也使得这些函数的const正确性稍微明显一些。如果你真的想加倍努力(并且你正在使用C ++ 11,我认为你就是这样),你也可以将这些函数标记为constexpr以使其成为可能在常量表达式中使用向量/点运算。