История про std::enable_shared_from_this
2026-03-10 22:38 Diff

Теги: c++, std::shared_ptr, с++11, класс, наследование, код, программирование, std::shared_ptr от this, указатель 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(); } }

Немного более подробное описание можно найти здесь.

Хотите узнать больше? Пишите в комментариях!