История про std::enable_shared_from_this
Представим, что вот прямо сейчас в текущей строчке кода некоего класса нужно получить std::shared_ptr от this, чтобы передать его куда-нибудь. Создавать или нет?
class SomeClass { public: using Ptr = std::shared_ptr<SomeClass>; Ptr createPtr() { return std::shared_ptr<SomeClass>{this}; } }
Вроде как правильный ответ – «нет». То есть не создавать. Потому что это чревато множеством проблем. Даже если всё сделать аккуратно. Что, если этот код уже вызывался и создаваемый сейчас std::shared_ptr будет уже вторым (третьим, четвертым…). Что произойдёт, когда придёт время этим указателям уничтожиться?
Правильно, многократное освобождение одной и той же памяти. Ведь все созданные экземпляры std::shared_ptr являются независимыми, хотя и ссылаются на одну и ту же память (по указателю this).
Что же делать?
Либо пересматривать дизайн, либо ознакомиться с новым (хотя уже не таким уж и новым) шаблонным классом, который присутствует, начиная с С++11 – встречайте, std::enable_shared_from_this.
Кроме претензии на звание самого длинного и непонятного названия, данный класс позволяет корректно создавать std::shared_ptr с использованием указателя this. Единственное требование – целевой класс должен быть наследником std::enabled_shared_from_this.
class SomeClass : public std::enable_shared_from_this<SomeClass> public: using Ptr = std::shared_ptr<SomeClass>; Ptr createPtr() { // Теперь корректно return shared_from_this(); } }
Немного более подробное описание можно найти здесь.
Хотите узнать больше? Пишите в комментариях!