История про 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();
        }
}

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

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