Как выполнить ковариантные типы возврата при возврате shared_ptr?
using namespace boost;
class A {};
class B : public A {};
class X {
virtual shared_ptr<A> foo();
};
class Y : public X {
virtual shared_ptr<B> foo();
};
Обратные типы не являются ковариантными (и, следовательно, они не являются законными), но они были бы, если бы я использовал исходные указатели. Какая общепринятая идиома работает над этим, если есть одна?
Ответы
Ответ 1
Я думаю, что решение принципиально невозможно, потому что ковариация зависит от арифметики указателя, которая несовместима с интеллектуальными указателями.
Когда Y::foo
возвращает shared_ptr<B>
динамическому вызывающему абоненту, перед тем, как использовать, он должен быть добавлен к shared_ptr<A>
. В вашем случае a B*
может (возможно) просто быть переинтерпретирован как A*
, но для множественного наследования вам потребуется некоторое волшебство, чтобы рассказать С++ о static_cast<A*>(shared_ptr<B>::get())
.
Ответ 2
Не напрямую, но вы можете подделать его, сделав виртуальные функции недоступными извне класса и завернув вызов виртуальной функции в не виртуальную функцию. Недостатком является то, что вам придется помнить о реализации этой функции-обертки для каждого производного класса. Но вы можете обойти это, добавив в макрос как объявление функции virtul, так и оболочку.
using namespace boost; // for shared_ptr, make_shared and static_pointer_cast.
// "Fake" implementation of the clone() function.
#define CLONE(MyType) \
shared_ptr<MyType> clone() \
{ \
shared_ptr<Base> res = clone_impl(); \
assert(dynamic_cast<MyType*>(res.get()) != 0); \
return static_pointer_cast<MyType>(res); \
}
class Base
{
protected:
// The actual implementation of the clone() function.
virtual shared_ptr<Base> clone_impl() { return make_shared<Base>(*this); }
public:
// non-virtual shared_ptr<Base> clone();
CLONE(Base)
};
class Derived : public Base
{
protected:
virtual shared_ptr<Base> clone_impl() { return make_shared<Derived>(*this); }
public:
// non-virtual shared_ptr<Derived> clone();
CLONE(Derived)
};
int main()
{
shared_ptr<Derived> p = make_shared<Derived>();
shared_ptr<Derived> clone = p->clone();
return 0;
}
Ответ 3
Я просто возвращаю пустой указатель и немедленно переношу его в общий указатель.