bad_weak_ptr при вызове shared_from_this () в базовом классе - PullRequest
4 голосов
/ 21 февраля 2012

У меня есть класс SuperParent, класс Parent (производный от SuperParent), и оба содержат класс от shared_ptr до Child (который содержит от weak_ptr до SuperParent). К сожалению, я получаю исключение bad_weak_ptr при попытке установить указатель Child. Код следующий:

#include <boost/enable_shared_from_this.hpp>
#include <boost/make_shared.hpp>
#include <boost/shared_ptr.hpp>
#include <boost/weak_ptr.hpp>

using namespace boost;

class SuperParent;

class Child {
public:
    void SetParent(shared_ptr<SuperParent> parent)
    {
        parent_ = parent;
    }
private:
    weak_ptr<SuperParent> parent_;
};

class SuperParent : public enable_shared_from_this<SuperParent> {
protected:
    void InformChild(shared_ptr<Child> grandson)
    {
        grandson->SetParent(shared_from_this());
        grandson_ = grandson;
    }
private:
    shared_ptr<Child> grandson_;
};

class Parent : public SuperParent, public enable_shared_from_this<Parent> {
public:
    void Init()
    {
        child_ = make_shared<Child>();
        InformChild(child_);
    }
private:
    shared_ptr<Child> child_;
};

int main()
{
    shared_ptr<Parent> parent = make_shared<Parent>();
    parent->Init();
    return 0;
}

1 Ответ

5 голосов
/ 21 февраля 2012

Это потому, что ваш родительский класс наследует enable_shared_from_this дважды. Вместо этого вы должны наследовать его один раз - через SuperParent. И если вы хотите иметь возможность получить shared_ptr в классе Parent, вы также можете наследовать его от следующего вспомогательного класса:

template<class Derived> 
class enable_shared_from_This
{
public:
typedef boost::shared_ptr<Derived> Ptr;

Ptr shared_from_This()
{
    return boost::static_pointer_cast<Derived>(static_cast<Derived *>(this)->shared_from_this());
}
Ptr shared_from_This() const
{
    return boost::static_pointer_cast<Derived>(static_cast<Derived *>(this)->shared_from_this());
}
};

Тогда

class Parent : public SuperParent, public enable_shared_from_This<Parent>
...