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

Курсы

Программирование
iOS Developer. Basic
-23%
Python Developer. Professional
-13%
Разработчик на Spring Framework
-23%
Golang Developer. Professional
-17%
Python Developer. Basic
-16%
iOS Developer. Professional
-13%
Node.js Developer
-15%
Unity Game Developer. Professional
-11%
React.js Developer
-12%
Android Developer. Professional
-7%
Software Architect
-12%
C++ Developer. Professional
-8%
Разработчик C#
-8%
Backend-разработчик на PHP
-8%
Архитектура и шаблоны проектирования
-12%
Программист С Базы данных Framework Laravel PostgreSQL Reverse-Engineering. Professional CI/CD Agile Project Manager Нереляционные базы данных Супер - интенсив по паттернам проектирования Супер-практикум по использованию и настройке GIT IoT-разработчик Advanced Fullstack JavaScript developer Супер-интенсив "Azure для разработчиков"
Инфраструктура
Мониторинг и логирование: Zabbix, Prometheus, ELK
-17%
DevOps практики и инструменты
-18%
Архитектор сетей
-21%
Инфраструктурная платформа на основе Kubernetes
-22%
Супер-интенсив «IaC Ansible»
-16%
Супер-интенсив по управлению миграциями (DBVC)
-16%
Administrator Linux. Professional
-5%
Administrator Linux.Basic
-10%
Супер-интенсив «ELK»
-10%
Базы данных Сетевой инженер AWS для разработчиков Cloud Solution Architecture Разработчик голосовых ассистентов и чат-ботов Внедрение и работа в DevSecOps Супер-практикум по работе с протоколом BGP Супер - интенсив по паттернам проектирования Супер - интенсив по Kubernetes Супер-интенсив «СУБД в высоконагруженных системах»
Специализации Курсы в разработке Подготовительные курсы
+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 в стандартной библиотеке", о котором говорилось на открытом уроке...

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