能否巧妙地使用虚拟继承让 CloneableMixin::clone 覆盖 Base::clone?
您的 CloneableMixin<Base,Derived> 不能覆盖 Base 的任何方法 - 要么
多态或隐藏 - 因为CloneableMixin<Base,Derived> 是
不是源自Base。
另一方面,如果CloneableMixin<Base,Derived> 是派生自Base
你不再需要它是一个 mixin,因为 -
class Derived : public CloneableMixin<Base,Derived> {....};
将继承Base。
因此,对于您的示例的需要,此处说明的解决方案就足够了:
#include <iostream>
// cloner v1.0
template <class Base, class Derived>
struct cloner : Base
{
Base *clone() const override {
return new Derived( dynamic_cast<const Derived &>(*this) );
}
~cloner() override {};
};
struct Base
{
virtual Base * clone() const = 0;
Base() {
std::cout << "Base()" << std::endl;
}
virtual ~Base() {
std::cout << "~Base()" << std::endl;
}
};
struct A : cloner<Base,A>
{
A() {
std::cout << "A()" << std::endl;
}
~A() override {
std::cout << "~A()" << std::endl;
}
};
int main()
{
A a;
Base * pb = a.clone();
delete pb;
}
(如果您正在编译为 C++03 标准而不是 C++11,那么您可以
只需删除出现的override 关键字。)
此解决方案将分解为一些更真实的类层次结构,
例如在这张Template Method Pattern 的插图中:
#include <iostream>
#include <memory>
using namespace std;
// cloner v1.0
template<class B, class D>
struct cloner : B
{
B *clone() const override {
return new D(dynamic_cast<D const&>(*this));
}
~cloner() override {}
};
/* Abstract base class `abstract` keeps the state for all derivatives
and has some pure virtual methods. It has some non-default
constructors.
*/
struct abstract
{
virtual ~abstract() {
cout << "~abstract()" << endl;
}
int get_state() const {
return _state;
}
void run() {
cout << "abstract::run()" << endl;
a_root_method();
another_root_method();
}
virtual void a_root_method() = 0;
virtual void another_root_method() = 0;
virtual abstract * clone() const = 0;
protected:
abstract()
: _state(0) {
cout << "abstract(): state = " << get_state() << endl;
}
explicit abstract(int state) : _state(state) {
cout << "abstract(" << state << ") : state = "
<< get_state() << endl;
}
int _state;
};
/* Concrete class `concrete` inherits `abstract`
and implements the pure virtual methods.
It echoes the constructors of `abstract`. Since `concrete`
is concrete, it requires cloneability.
*/
struct concrete : cloner<abstract,concrete>
{
concrete() {
cout << "concrete(): state = " << get_state() << endl;
}
explicit concrete(int state) : abstract(state) { //<- Barf!
cout << "concrete(" << state << ") : state = "
<< get_state() << endl;
}
~concrete() override {
cout << "~concrete()" << endl;
}
void a_root_method() override {
++_state;
cout << "concrete::a_root_method() : state = "
<< get_state() << endl;
}
void another_root_method() override {
--_state;
cout << "concrete::another_root_method() : state = "
<< get_state() << endl;
}
};
int main(int argc, char **argv)
{
concrete c1;
unique_ptr<abstract> pr(new concrete(c1));
pr->a_root_method();
pr->another_root_method();
unique_ptr<abstract> pr1(pr->clone());
pr1->a_root_method();
return 0;
}
当我们尝试构建它时,编译器会在
concrete 的构造函数中的初始化 abstract(state)(在 Barf!
评论),说:
error: type 'abstract' is not a direct or virtual base of 'concrete'
或类似的词。确实,concrete 的直接基数不是abstract
但是cloner<abstract,concrete>。但是,我们不能将构造函数重写为:
/*Plan B*/ explicit concrete(int state) : cloner<abstract,concrete>(state){....}
因为没有像这样的构造函数
cloner<abstract,concrete>::cloner<abstract,concrete>(int)
但是编译器的诊断建议修复。 这个是虚拟的
继承可以提供帮助。我们需要abstract 成为concrete 的虚拟基础,这
实际上意味着“concrete的名誉直属基地”,我们可以做到这一点
只需将B 设为cloner<B,D> 的虚拟 基础:
// cloner v1.1
template<class B, class D>
struct cloner : virtual B
{
B *clone() const override {
return new D(dynamic_cast<D const&>(*this));
}
~cloner() override {}
};
这样,我们就有了一个干净的构建和输出:
abstract(): state = 0
concrete(): state = 0
concrete::a_root_method() : state = 1
concrete::another_root_method() : state = 0
concrete::a_root_method() : state = 1
~concrete()
~abstract()
~concrete()
~abstract()
~concrete()
~abstract()
原则上对虚拟继承持谨慎态度是有充分理由的
并至少在它具有架构的情况下保留它的使用
基本原理 - 不是为了解决方法,因为我们刚刚使用了它。
如果我们更愿意在这个问题上不使用虚拟继承,那么我们
必须以某种方式确保存在cloner<B,D> 的构造函数
回显B 的any 构造函数,用于任意B。那么任何对应的
D 的构造函数将能够初始化其直接基 cloner<B,D>
无论论据是什么。
这是 C++03 的白日梦,但具有可变参数模板的魔力
C++11 中的参数很简单:
// cloner v1.2
template<class B, class D>
struct cloner : B
{
B *clone() const override {
return new D(dynamic_cast<D const&>(*this));
}
~cloner() override {}
// "All purpose constructor"
template<typename... Args>
explicit cloner(Args... args)
: B(args...){}
};
有了这个,我们可以将concrete构造函数重写为/*Plan B*/,并且
再次,我们有一个正确的构建和可执行文件。