Ceci est une ancienne révision du document !
Table des matières
Durée de vie
Lors d'un reset, le nouvel objet commence par être construit avant de détruire le précédent. Si les objets ont une place importante en mémoire, il peut être intéressant de commencer par faire un reset() avant de faire le reset(…).
#include <iostream> #include <memory> class A { public: A() { std::cout << "Constructor\n"; } ~A() { std::cout << "Destructor\n"; } }; int main() { std::shared_ptr<A> a = std::make_shared<A>(); a.reset(new A()); a = std::make_shared<A>(); }
Sortie :
Constructor Constructor Destructor Constructor Destructor Destructor
std::weak_ptr en tant que clé dans un conteneur
Il faut utiliser le comparateur owner_less.
std::map<U, T, std::owner_less<U>> destination;
Préserver l'attribut const sur les pointeurs
Il faut utiliser std::propagate_const<std::unique_ptr<X>>.
Il existe aussi indirect_value qui peut se substituer à std::propagate_const<std::unique_ptr<X>>. Attention, indirect_value ne permet pas de se substituer à std::propagate_const<std::shared_ptr<X>> (voir Can I replace shared_ptr by indirect_value ? Archive du 04/08/2022 le 18/10/2022).
Soit la classe commune X :
struct X { void g() const { std::cout << "g (const)\n"; } void g() { std::cout << "g (non-const)\n"; } };
et le programme commun :
int main() { Y y; y.f(); const Y cy; cy.f(); }
- Cas avec la classe
Xintégrée dans une classeYavec pointeur classique
struct Y { Y() { } void f() const { std::cout << "f (const)\n"; m_ptrX.g(); } void f() { std::cout << "f (non-const)\n"; m_ptrX.g(); } // Ici, X est une instance sans pointeur. X m_ptrX; };
Rendu sans surprise :
f (non-const) g (non-const) f (const) g (const)
Si la méthode f const est appelée, la méthode g const est appelée également car une méthode const ne peut pas appeler une méthode non const.
- Cas avec un pointeur intelligent
struct Y { Y() : m_ptrX{} { } void f() const { std::cout << "f (const)\n"; m_ptrX->g(); } void f() { std::cout << "f (non-const)\n"; m_ptrX->g(); } std::unique_ptr<X> m_ptrX; };
Rendu :
f (non-const) g (non-const) f (const) g (non-const)
On ne respecte l'obligation d'appeler des méthodes const depuis une méthode const.
- Cas avec un pointeur unique et propagate_const
Prérequis : minimum clang 3.9, gcc 6.1. Pas dans Visual Studio 2017 sauf sur GitHub.
#include <experimental/propagate_const> std::experimental::propagate_const<std::unique_ptr<X>> m_ptrX;
Rendu :
f (non-const) g (non-const) f (const) g (const)
