История про std::enable_shared_from_this | OTUS
⚡ Открываем подписку на курсы!
Проходите параллельно 3 онлайн-курса в месяц по цене одного.
Подробнее

Курсы

Программирование
iOS Разработчик. Продвинутый курс Программист 1С Реверс-инжиниринг. Продвинутый курс
-16%
Java Developer. Professional
-17%
JavaScript Developer. Professional
-18%
Flutter Mobile Developer
-15%
MS SQL Server Developer
-14%
Unity Game Developer. Basic
-19%
Супер - практикум по использованию и настройке GIT
-18%
Супер-интенсив "СУБД в высоконагруженных системах"
-18%
Web-разработчик на Python
-11%
Backend-разработчик на PHP
-8%
PostgreSQL
-10%
Базы данных
-19%
Android-разработчик. Базовый курс Разработчик Python. Продвинутый курс Разработчик на Spring Framework AWS для разработчиков Cloud Solution Architecture CI/CD Vue.js разработчик Разработчик Node.js Scala-разработчик Супер - интенсив по Kubernetes Symfony Framework Advanced Fullstack JavaScript developer
Специализации Курсы в разработке Подготовительные курсы
+7 499 938-92-02

История про std::enable_shared_from_this

C___Deep_12-5020-6fd9db.09_site.png

Представим, что вот прямо сейчас в текущей строчке кода некоего класса нужно получить 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();
        }
}

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

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

Не пропустите новые полезные статьи!

Спасибо за подписку!

Мы отправили вам письмо для подтверждения вашего email.
С уважением, OTUS!

Автор
1 комментарий
0

Так вот он какой "пример CRTP в стандартной библиотеке", о котором говорилось на открытом уроке...

Для комментирования необходимо авторизоваться