模板类中对 operator+ 的未定义引用

Undefined reference to operator+ in template class

提问人: 提问时间:3/1/2015 更新时间:3/1/2015 访问量:540

问:

我正在尝试为游戏制作一个简单的 2D 坐标向量类,但在覆盖 operator+ 时遇到了问题。向量使用模板来设置其 X 和 Y 值的数据类型,因此所有函数(包括运算符重载)都必须使用该模板。所有成员函数都工作正常,但非成员函数给我一个错误。

这是错误:

In function `ZNK13AxisAlignedBB8maxPointEv':
undefined reference to `Vector<double> operator+<double>(Vector<double> const&, Vector<double> const&)'

这是给我错误的一段代码:

Vector2d AxisAlignedBB::maxPoint() const {
    return position + dimensions;
    //position and dimensions are both type Vector<double>
}

以下是函数的声明:

template <typename T> class Vector {
    //class definition...
}


template<typename T>
Vector<T> operator+(const Vector<T>&, const Vector<T>&);

下面是该函数的实现。据我所知,它应该与上面的声明相匹配:

template<typename T> Vector<T> operator+(const Vector<T>& v1, const Vector<T>& v2){
    return v1.add(v2); //returns Vector<T>(v1.x + v2.x, v1.y + v2.y)
}

我也一直在使用 SFML 作为游戏中的图形,所以为了确保这一点,我查看了 SFML/System/Vector2.hpp,它的工作与我的矢量基本相同。这是同一函数的版本,除了参数名称外,它看起来与我的相同:

template <typename T>
Vector2<T> operator +(const Vector2<T>& left, const Vector2<T>& right);

我无权访问 SFML 的函数实现,因此我无法检查以确保它相同。我一开始不使用 SFML 向量的原因是因为它们不包括量级、叉积等函数。另外,我真的很懒,不想把“sf::”放在所有东西的前面。

编辑:Vector.h / .cpp的全文

向量.h:

#ifndef VECTOR_H
#define VECTOR_H

#include <SFML/System/Vector2.hpp>

template <typename T> class Vector {
public:
    T x;
    T y;

    Vector<T>() : x(0), y(0) {};

    Vector<T>(T ax, T ay) : x(ax), y(ay) {};

    ///Conversion from SFML vector of same type
    Vector<T>(const sf::Vector2<T>& v) : x(v.x), y(v.y) {};

    /**
     * Returns the magnitude of this Vector<T>.
     */
    T mag() const;

    /**
     * Returns the square of the magnitude of this Vector<T>, cutting
     * out an expensive sqrt() call.
     */
    T magSquared() const;

    /**
     * Returns the dot product of this and the given Vector<T>.
     */
    T dot(const Vector<T>&) const;

    /**
     * Returns the distance between this and the given Vector<T>.
     */
    T distance(const Vector<T>&) const;

    /**
     * Returns the square of the distance between this and the given Vector<T>,
     * cutting out an expensive sqrt() call.
     */
    T distanceSquared(const Vector<T>&) const;

    /**
     * Returns a new Vector<T> with the same coordinates.
     */
    Vector<T> clone() const;

    /**
     * Returns the resultant Vector<T> of adding the given Vector<T>.
     */
    Vector<T> add(const Vector<T>&) const;

    /**
     * Returns the resulting Vector<T> of subtracting the given Vector<T>.
     */
    Vector<T> sub(const Vector<T>&) const;

    /**
     * Returns the resulting Vector<T> of multiplying this Vector<T> by
     * the given scalar value.
     */
    Vector<T> mult(T) const;

    /**
     * Returns the projection of this Vector<T> onto the given axis.
     * In this function, this Vector<T> is the point to be projected
     * and the given Vector<T> is the axis.
     */
    Vector<T> proj(const Vector<T>&) const;

    /**
     * Returns the projection of the given Vector<T> on to this axis.
     * In this function, this Vector<T> acts as the axis and the given
     * Vector<T> is the point to be projected.
     */
    Vector<T> proj2(const Vector<T>&) const;

    /**
     * Revolves this Vector<T> about the origin by the given angle in radians.
     */
    Vector<T> revolve(T) const;

    /**
     * Revolves this Vector<T> about the given Vector<T>
     * by the given angle in radians.
     */
    Vector<T> revolveAbout(T, const Vector<T>&) const;

    /**
     * Returns the normalized version of this Vector<T>.
     */
    Vector<T> normalize() const;

    /**
     * Performs a 2D cross product with the given Vector<T>.
     */
    T crossVector(const Vector<T>&) const;

    /**
     * Performs a 2D cross product with the given scalar value.
     * This function performs the operation this X value.
     */
    Vector<T> crossScalar(T) const;

    /**
     * Performs a 2D cross product with the given scalar value.
     * This function performs the operation value X this.
     */
    Vector<T> crossScalar2(T) const;

    Vector<T>& operator+=(const Vector<T>& v){
        x += v.x;
        y += v.y;
        return *this;
    }

    Vector<T>& operator-=(const Vector<T>& v){
        x -= v.x;
        y -= v.y;
        return *this;
    }

    Vector<T>& operator*=(const T t){
        x *= t;
        y *= t;
        return *this;
    }

    Vector<T>& operator/=(const T t){
        x /= t;
        y /= t;
        return *this;
    }

    Vector<T> operator-(){
        return Vector(-x, -y);
    }

    ///Conversion from SFML vector of same type
    Vector<T>& operator=(const sf::Vector2<T>&);

    ///Conversion to SFML vector of same type
    operator sf::Vector2<T>();

protected:
private:
};

typedef Vector<double> Vector2d;
typedef Vector<float> Vector2f;
typedef Vector<int> Vector2i;

template<typename T> Vector<T> operator+(const Vector<T>& v1, const Vector<T>& v2){
    return Vector<T>(v1.x + v2.x, v1.y + v2.y);
}

template<typename T> Vector<T> operator-(const Vector<T>& v1, const Vector<T>& v2){
    return Vector<T>(v1.x - v2.x, v1.y - v2.y);
}

template<typename T> Vector<T> operator*(const Vector<T>& v, const T t){
    return Vector<T>(v.x * t, v.y * t);
}

template<typename T> Vector<T> operator*(const T t, const Vector<T>& v){
    return Vector<T>(v.x * t, v.y * t);
}

template<typename T> Vector<T> operator/(const Vector<T>& v, const T t){
    return Vector<T>(v.x / t, v.y / t);
}

#endif // VECTOR_H

矢量.cpp:

#include "Vector.h"
#include <cmath>

template<typename T> T Vector<T>::mag() const {
    return std::sqrt(magSquared());
}

template<typename T> T Vector<T>::magSquared() const {
    return std::pow(x, 2) + std::pow(y, 2);
}

template<typename T> T Vector<T>::dot(const Vector<T>& v) const {
    return (v.x * x) + (v.y * y);
}

template<typename T> T Vector<T>::distanceSquared(const Vector<T>& v) const {
    return sub(v).magSquared();
}

template<typename T> T Vector<T>::distance(const Vector<T>& v) const {
    return sub(v).mag();
}

template<typename T> Vector<T> Vector<T>::clone() const {
    return Vector(x, y);
}

template<typename T> Vector<T> Vector<T>::add(const Vector<T>& v) const {
    return Vector(x + v.x, y + v.y);
}

template<typename T> Vector<T> Vector<T>::sub(const Vector<T>& v) const {
    return Vector(x - v.x, y - v.y);
}

template<typename T> Vector<T> Vector<T>::mult(T t) const {
    return Vector(t * x, t * y);
}

template<typename T> Vector<T> Vector<T>::proj(const Vector<T>& axis) const {
    return axis.proj2(*this);
}

template<typename T> Vector<T> Vector<T>::proj2(const Vector<T>& point) const {
    T p = dot(point) / magSquared();
    return mult(p);
}

template<typename T> Vector<T> Vector<T>::revolve(T a) const {
    a += std::atan2(y, x);
    return Vector(mag() * std::cos(a), mag() * std::sin(a));
}

template<typename T> Vector<T> Vector<T>::revolveAbout(T a, const Vector<T>& v) const {
    return sub(v).revolve(a).add(v);
}

template<typename T> Vector<T> Vector<T>::normalize() const {
    return Vector(x / mag(), y / mag());
}

template<typename T> T Vector<T>::crossVector(const Vector<T>& v) const{
    return x * v.y - y * v.x;
}

template<typename T> Vector<T> Vector<T>::crossScalar(T t) const {
    return Vector(t * y, -t * x);
}

template<typename T> Vector<T> Vector<T>::crossScalar2(T t) const {
    return crossScalar(-t);
}

template<typename T> Vector<T>& Vector<T>::operator=(const sf::Vector2<T>& v){
    x = v.x;
    y = v.y;
    return *this;
}

template<typename T> Vector<T>::operator sf::Vector2<T>(){
    return sf::Vector2<T>(x, y);
}
C++ 模板 operator-keyword undefined-reference

评论

2赞 delphifirst 3/1/2015
您是否也将函数的实现放入了标头中?
1赞 M.M 3/1/2015
可能重复 为什么模板只能在头文件中实现?
0赞 3/1/2015
我本来没有,但我在看完这个问题后移动了它:链接。这样做解决了第一个问题,但现在我也在非运算符函数中遇到了同样的错误。
0赞 3/1/2015
add() 的数据类型为 const 并返回 Vector 的新实例。
1赞 M.M 3/1/2015
运算符函数和非运算符函数之间除了名称之外没有区别。您是否将模板函数的所有实现都放在标头中,并确保它们从调用函数的位置可见?如果您仍然遇到问题,那么您将不得不发布显示问题的代码,因为试图描述代码太模糊了

答: 暂无答案