模板层次结构中的可选虚拟函数,具体取决于参数
optional virtual function in template hierarchy depending on parameter
我有一个模板层次结构,我希望它具有clone()
功能,具体取决于模板类型是否可复制构造。作为第一步,我想从其他参数bool Clonable
开始:
template<class T, bool Clonable>
class Base {
T t;
void foo();
virtual void bar();
virtual unique_ptr<Base> clone() const = 0; //should be only for Clonable=true
};
template<class T, bool Clonable>
class Derived : public Base<T, Clonable> {
virtual void bar() override;
virtual unique_ptr<Base> clone() const override; ////should be only for Clonable=true
};
不幸的是,虚函数的实例化并不取决于它们是否被调用。所以我想我应该选择部分专业化。但是,直接的方法会导致大量的代码重复。谁能推荐以最少的代码重复实现这一目标的方法?
不幸的是,与这里的一些评论相反,SFINAE 在这里无法为您提供帮助。这是因为模板类的非模板成员不被视为模板,因此不能被 SFINAE 输出:http://coliru.stacked-crooked.com/a/258e20a0293d93f0。解决此问题的标准方法通常是以微不足道的方式使其成为模板:
template <class U = T, std::enable_if ... >
virtual std::unique_ptr<Base> clone() const = 0;
但是虚拟函数不能是模板,所以这不起作用。
在这种情况下,避免重复的方法是从有条件地具有成员的类继承:
template <class Base, bool Cloneable>
struct CloneInterface;
template <class Base>
struct CloneInterface<Base, false> {};
template <class Base>
struct CloneInterface<Base, true> {
virtual unique_ptr<Base> clone() const = 0;
}
现在你只是继承:
template<class T, bool Clonable>
class Base : CloneInterface<Base<T, Clonable>, Clonable> {
T t;
void foo();
virtual void bar();
};
请注意,我们继承自在派生上模板化的基类(有问题的派生类称为Base
,以使事情更加混乱:-))。这种技术称为CRTP,它非常强大,因为它可以将接口和实现注入到类中,并且如您所见,也可以有条件地这样做。
为了获得实现,我们再次使用 CRTP:
template <class T, bool Clonable, class D>
struct BaseHelper;
template <class T, class D>
struct BaseHelper<T, false, D> : Base<T, false> {};
template <class T, class D>
struct BaseHelper<T, true, D> : Base<T, true> {
unique_ptr<Base<T, true>> clone() override { return make_unique<D>(static_cast<D&>(*this)); }
};
template<class T, bool Clonable>
class Derived : public BaseHelper<T, Clonable, Derived<T, Clonable>> {
virtual void bar() override;
};
相关文章:
- static_assert在宏中,但也可以扩展到可以用作函数参数的东西
- C++中的高效循环缓冲区,它将被传递给C样式数组函数参数
- 当从函数参数中的临时值调用复制构造函数时
- 如何从"decltype()"获取函数参数的数量<funtion>?
- 如何将lambda作为模板类的成员函数参数
- 模板参数推导失败,函数参数/参数不匹配
- 如何在C++中将迭代器作为函数参数传递
- 将函数参数"const char*"转换为"std::string_view"是
- C++ 如何将数组值解压缩为函数参数
- 主函数参数的属性
- 具有两个间接寻址运算符 (C++) 的函数参数的用途
- "Warning: Comma within array index expression"但逗号分隔函数参数
- 如何定义在用作函数参数时工作的类模板的转换
- 将函数参数完美转发到函数指针:按值传递呢?
- 为什么我不能将引用作为 std::async 的函数参数传递
- 什么..(省略号)作为函数原型中唯一的函数参数,C++?
- 是否可以就地构造一个固定大小的数组作为函数参数?
- 接受模板作为函数参数
- 将成员函数作为构造函数参数调用时出错 "Variable is not a type name"
- Arduino 函数参数