3

我正在尝试根据模板参数指定的 2D 或 3D 专门化一些几何函数。如果我为问题的玩具版本包含一些(非常损坏的)代码,则最好:

template <typename T, int d>
class Point
{
public:
    int x;
    int y;
    int z;

    T add ()
    {
        return T(0);
    }

    template <>
    T add <T, 2> ()
    {
        return x + y;
    }

    template <>
    T add <T, 3> ()
    {
        return x + y + z;
    }
};

这段代码编译失败。我已经尝试了很多不同的模板参数格式和类定义的组合,但找不到一种方法来对 'd' 进行函数专业化,同时保留 'T' 通用。

在我的实际解决方案中,我试图计算梯度、曲率、插值等,专门用于 2D 或 3D 情况。有些事情,比如梯度计算,可以简单地使用 'd' 参数来限制 for 循环迭代。其他的,如插值,需要单独的 2D 和 3D 函数。

非常感谢任何提示!

4

2 回答 2

7

我会建议这个解决方案:

template <typename T, int d> 
class Point : public Point<T, d-1>
{
   typedef Point<T, d-1> base;
   T m_value;
public:
    T add()
    {
        return m_value + base::add();
    }   

    //another method which returns you the point value
    template<int N>
    T get()
    {
       return N==d ? m_value : base::get<N>();
    }
};

template <typename T>
class Point<T,0>
{
protected:
    T add()
    {
        return T(); //default value which is zero for all builtin types
    }
    template<int N>
    T get() { return T(); }

};

使用此解决方案,您可以拥有任意多的点,但大于zero.

Point<int,1> p1;  //contains 1 point
Point<int,2> p2;  //contains 2 points
Point<int,3> p3;  //contains 3 points
Point<int,4> p4;  //contains 4 points
Point<int,5> p5;  //contains 5 points

auto x1 = p5.get<1>(); //get first point
auto x3 = p5.get<3>(); //get third point
auto x4 = p5.get<4>(); //get fourth point

或者为了方便起见,使用这些 typedef:

typedef Point<int,2> Point2D;
typedef Point<int,3> Point3D;

//then use them
Point2D p2d;
Point3D p3d;

嗯,这只是一个可以进一步增强的基本概念,支持许多有用的功能。我只是写get<>来演示一个似乎有用的功能。

于 2012-12-31T05:22:37.570 回答
5

这是您的示例的工作方式。您首先声明您的主模板:

template <typename T, int d> class Point;

无需定义它,因为您没有通用实现。

接下来,您为不同数量的维度创建偏特化,但您的偏特化仍将类型 T 作为模板参数:

template <typename T>
class Point<T,2>
{
public:
    T x;
    T y;

    T add()
    {
        return x + y;
    }
};

template <typename T>
class Point<T,3>
{
public:
    T x;
    T y;
    T z;

    T add()
    {
        return x + y + z;
    }
};
于 2012-12-31T05:16:07.087 回答