我有一个类模板Shape ,其中包含有关某些形状(可以是三维或二维)的信息。 我只希望提供一些预定义的形状(立方体,球形和正方形)。 所有这些预定义的形状都具有相同的属性(因此,多维数据集始终具有相同的体积,我只需要记住一个多维数据集的属性)。 为了禁止某人创建其他Shape ,我将构造函数private

// Flag for the possible shapes
enum class Tag
{
    SPHERE,
    CUBE,
    SQUARE
};

template<std::size_t N>
class Shape
{
public:
    // Predefined shapes.
    static const Shape<3> SPHERE;
    static const Shape<3> CUBE;
    static const Shape<2> SQUARE;
    // Information stored about the given shapes
    const Tag tag; // tag specifying the shape
    const double v; // Shape volume/area
    const std::array<double, 2*N> surrounding_box; // Storing intervals for a surrounding box
    //... Some other information that depends on template parameter N
private:
    // Private constructor. This prevents other, unintended shapes from being created
    Shape(Tag tag, double v, const std::array<double, 2*N> surrounding_box):
            tag{tag}, v {v}, surrounding_box {surrounding_box} {};
};

// Initialization of predefined shape: SPHERE
template<std::size_t N>
const Shape<3> Shape<N>::SPHERE(Tag::SPHERE, 3.0,{{0.0,2.7,0.0,2.7,0.0,2.7}});

// Initialization of predefined shape: CUBE
template<std::size_t N>
const Shape<3> Shape<N>::CUBE(Tag::CUBE, 1.0,{{0.0,1.0,0.0,1.0,0.0,1.0}});

// Initialization of predefined shape: SQUARE
template<std::size_t N>
const Shape<2> Shape<N>::SQUARE(Tag::SQUARE, 1.0,{{0.0,1.0,0.0,1.0}});

现在我可以得到一个多维数据集为:

Shape<3> cube = Shape<3>::CUBE;

这似乎工作正常。

当我想让Shape实例作为另一个类模板Object成员时出现问题。 具体来说,我无法为Object类模板编写可正常工作的构造函数:

template <std::size_t N>
class Object
{
public:
    Object(Tag shape_tag, double weight, double elevation):
            weight {weight}, elevation {elevation}
    {
        switch(shape_tag)
        {
            case Tag::CUBE:
            {
                shape = Shape<3>::CUBE;
                break;
            }
            case Tag::SPHERE:
            {
                shape = Shape<3>::SPHERE;
                break;
            }
            case Tag::SQUARE:
            {
                shape = Shape<2>::SQUARE;
                break;
            }
        }
    }
private:
    Shape<N> shape;
    double weight;
    double elevation;
};

创建一个Object

Object<3> object(Tag::CUBE, 1.0,1.0);

因编译器错误error: no matching function for call to 'Shape<3ul>::Shape()'失败error: no matching function for call to 'Shape<3ul>::Shape()' 我认为,因为我没有为shape使用初始化列表,所以Object的构造函数尝试调用默认的构造函数Shape() ,该构造函数不可用。 我还尝试将构造的Shape部分移至单独的初始化函数,然后可以在初始化列表中调用它。 但是,在那种情况下,模板部分不断产生不同的问题(因为我需要能够初始化Shape<2>Shape<3>对象)。

我该如何解决这个问题? 还是有更好的方法来确保只有一些预定义的Shape可用,而不将其构造函数设为私有?

PS。 此处介绍的形状和对象问题仅仅是MWE。

#1楼 票数:4 已采纳

创建工厂:

template <std::size_t N> Shape<N> MakeShape(Tag shape_tag);

template <>
Shape<3> MakeShape(Tag shape_tag)
{
    switch(shape_tag)
    {
        case Tag::CUBE: return Shape<3>::CUBE;
        case Tag::SPHERE: return Shape<3>::SPHERE;
    }
    throw std::runtime_error("Invalid tag");
}

template <>
Shape<2> MakeShape(Tag shape_tag)
{
    switch(shape_tag)
    {
        case Tag::SQUARE: return Shape<3>::SQUARE;
    }
    throw std::runtime_error("Invalid tag");
}

接着

template <std::size_t N>
class Object
{
public:
    Object(Tag shape_tag, double weight, double elevation):
shape{MakeShape<N>(shape_tag)}, weight {weight}, elevation {elevation}
    {
    }
};

  ask by JorenV translate from so

未解决问题?本站智能推荐:

1回复

如何检查模板参数是否为结构/类?

对于这样的小示例,如果T是struct/class ,我只想接受T ,并且拒绝诸如“ int”,“ char”,“ bool”等内置类型。
1回复

使用c ++ 17编译时无法从基类访问成员类型[重复]

这个问题在这里已经有了答案: 为什么我必须通过 this 指针访问模板基类成员? (3 个回答) 去年关闭。
1回复

如何构造具有相同参数的嵌套类模板的对象?

如何构造具有相同参数的嵌套类模板的对象? 如何编写构造函数来编译以下代码? 编译错误:
1回复

模板类模板构造器专业化

我有一个模板类,成员的类型取决于类的模板参数。 该类具有模板构造函数。 我如何专门针对类的模板参数的不同情况确定构造函数,以决定所述成员的类型。 有问题的不同类型是具有不同构造函数签名的类,我想在初始化列表中调用成员的构造函数。 有解决方法吗? 我想避免诉诸于工厂函数,因为我不想依靠便
1回复

使用聚合初始化程序初始化类'模板(聚合类型)成员但没有额外的括号

有这个代码: 创建myProperty类型对象时: 是否有一种优雅的方式使vp2初始化工作? 专门针对Vec3 myProperty是一种矫枉过正。
1回复

类模板实例仅限于预定义对象

我想要一个类模板template<std::size_t N> Shape ,其中模板参数N表示Shape的尺寸。 预定义Shape的数量应有限,例如Shape<2> SQUARE , Shape<3> CUBE和Shape<3> SPHERE
3回复

C++ - 从另一个类构造函数调用类构造函数

所以我正在学习 C++ 课程,在一次练习中,我需要在名为“Character”的类的构造函数中使用名为“Weapon”的类的构造函数,但是每次我尝试编译代码时,它都会给我这个错误: 即使我在 Weapon.h 中声明了这个确切的构造函数: 顺便说一句,我是法国人,这是我第一次发帖,但我尽力将
2回复

类模板成员专业化

我正在像这样的头文件中专门研究模板类的成员函数: 将专门化放在头文件中(没有内联)是否正确,还是应该在cpp文件中? 如上所示,它可以正常编译(使用VS2012),但是我很惊讶我没有遇到多个定义链接器错误。