This is an old revision of this page, as edited by AnomieBOT (talk | contribs) at 12:15, 18 May 2015 (Dating maintenance tags: {{Citation needed span}}). The present address (URL) is a permanent link to this revision, which may differ significantly from the current revision.
Revision as of 12:15, 18 May 2015 by AnomieBOT (talk | contribs) (Dating maintenance tags: {{Citation needed span}})(diff) ← Previous revision | Latest revision (diff) | Newer revision → (diff)In computer science, a smart pointer is an abstract data type that simulates a pointer while providing additional features, such as automatic memory management or bounds checking. These additional features are intended to reduce bugs caused by the misuse of pointers while retaining efficiency. Smart pointers typically keep track of the memory they point to. They may also be used to manage other resources, such as network connections and file handles.
Misuse of pointers is a major source of bugs. Smart pointers prevent most situations of memory leaks by making the memory deallocation automatic. More generally, they make object destruction automatic: the object controlled by a smart pointer is automatically destroyed (finalized and then deallocated) when the last (or only) owner of the object is destroyed, for example because the owner is a local variable, and execution leaves the variable's scope. Smart pointers also eliminate dangling pointers by postponing destruction until the object is no longer in use.
Several types of smart pointers exist. Some work with reference counting, others by assigning ownership of the object to a single pointer. If a language supports automatic garbage collection (for instance, Java or C#), then smart pointers are not needed for the reclamation and safety aspects of memory management, but are nevertheless useful for other purposes, such as cache data structure residence management and resource management of objects such as file handles or network sockets.
C++ smart pointers
In C++, a smart pointer is implemented as a template class that mimics, by means of operator overloading, the behaviors of a traditional (raw) pointer, (e.g. dereferencing, assignment) while providing additional memory management features.
Smart pointers can facilitate intentional programming by expressing in the type itself how the memory of the referent of the pointer will be managed. For example, if a C++ function returns a pointer, there is no way to know whether the caller should delete the memory of the referent when the caller is finished with the information.
some_type* ambiguous_function(); // What should be done with the result?
Traditionally, comments have been used to resolve the ambiguity, which is an error-prone, labor-intensive approach. C++11 introduced a way to ensure correct memory management in this case by declaring the function to return a unique_ptr
,
unique_ptr<some_type> obvious_function1();
The declaration of the function return type as a unique_ptr makes explicit the fact that the caller takes ownership of the result, and the C++ runtime ensures that the memory for *some_type will be reclaimed automatically. Prior to C++11, unique_ptr can be replaced with auto_ptr.
Creating new objects
To ease the allocation of a
std::shared_ptr<some_type>
C++11 introduced:
auto s = std::make_shared<some_type>(constructor, parameters, here);
and similarly
std::unique_ptr<some_type>
Since C++14 one can use:
auto u = std::make_unique<some_type>(constructor, parameters, here);
It is preferred, in almost all circumstances, to use these facilities over the new
keyword:
unique_ptr
C++11 introduces std::unique_ptr
, defined in the header <memory>
.
A unique_ptr
is a container for a raw pointer, which the unique_ptr
is said to own. A unique_ptr
explicitly prevents copying of its contained pointer (as would happen with normal assignment), but the std::move
function can be used to transfer ownership of the contained pointer to another unique_ptr
. A unique_ptr
cannot be copied because its copy constructor and assignment operators are explicitly deleted.
std::unique_ptr<int> p1(new int(5)); std::unique_ptr<int> p2 = p1; //Compile error. std::unique_ptr<int> p3 = std::move(p1); //Transfers ownership. p3 now owns the memory and p1 is rendered invalid. p3.reset(); //Deletes the memory. p1.reset(); //Does nothing.
History
std::auto_ptr
is still available, but it is deprecated under C++11. The copy constructor and assignment operators of auto_ptr
do not actually copy the stored pointer. Instead, they transfer it, leaving the previous auto_ptr
object empty. This was one way to implement strict ownership, so that only one auto_ptr
object could own the pointer at any given time. This means that auto_ptr
should not be used where copy semantics are needed. Since auto_ptr
already existed with its copy semantics, it could not be upgraded to be a move-only pointer without breaking backwards compatibility with existing code
shared_ptr and weak_ptr
C++11 introduces std::shared_ptr
and std::weak_ptr
, defined in the header <memory>
.
A shared_ptr
is a container for a raw pointer. It maintains reference-counted ownership of its contained pointer in cooperation with all copies of the shared_ptr
. The object referenced by the contained raw pointer will be destroyed when and only when all copies of the shared_ptr
have been destroyed.
std::shared_ptr<int> p1(new int(5)); std::shared_ptr<int> p2 = p1; //Both now own the memory. p1.reset(); //Memory still exists, due to p2. p2.reset(); //Deletes the memory, since no one else owns the memory.
A weak_ptr
is a container for a raw pointer. It is created as a copy of a shared_ptr
. The existence or destruction of weak_ptr
copies of a shared_ptr
have no effect on the shared_ptr
or its other copies. After all copies of a shared_ptr
have been destroyed, all weak_ptr
copies become empty.
std::shared_ptr<int> p1(new int(5)); std::weak_ptr<int> wp1 = p1; //p1 owns the memory. { std::shared_ptr<int> p2 = wp1.lock(); //Now p1 and p2 own the memory. if(p2) // As p2 is initialized from a weak pointer, you have to check if the memory still exists! { //Do something with p2 } } //p2 is destroyed. Memory is owned by p1. p1.reset(); //Memory is deleted. std::shared_ptr<int> p3 = wp1.lock(); //Memory is gone, so we get an empty shared_ptr. if(p3) { //Will not execute this. }
Circular references
Because the implementation of shared_ptr
uses reference counting, circular references are potentially a problem. A circular shared_ptr
chain can be broken by changing the code so that one of the references is a weak_ptr
.
Concurrency guarantees
Multiple threads can safely simultaneously access different shared_ptr
and weak_ptr
objects that point to the same object.
The referenced object itself needs to be protected separately to ensure thread safety.
History
shared_ptr
and weak_ptr
are based on versions used by the Boost libraries. TR1 first introduced them to the standard, but C++11 gives them additional functionality in line with the Boost version.
See also
- Resource Acquisition Is Initialization (RAII)
- auto_ptr
- Opaque pointer
- Reference
- The Boost library includes boost::intrusive_ptr, a reference-counting and intrusively counted smart pointer implementation for C++.
References
- Sutter, Herb (20 April 2013). "Trip Report: ISO C++ Spring 2013 Meeting". isocpp.org. Retrieved 14 June 2013.
- ^ ISO 14882:2011 20.7.1
- CERT C++ Secure Coding Standard
- boost::shared_ptr thread safety (does not formally cover std::shared_ptr, but is believed to have the same threading limitations)
- http://www.boost.org/doc/libs/1_52_0/libs/smart_ptr/intrusive_ptr.html
External links
- Sample chapter "Smart Pointers" from the book Modern C++ Design: Generic Programming and Design Patterns Applied by Andrei Alexandrescu, Addison-Wesley, 2001.
- Code example "countptr.hpp" from the book The C++ Standard Library - A Tutorial and Reference by Nicolai M. Josuttis
- "Boost Smart Pointers"
- Article "The New C++: Smart(er) Pointers" by Herb Sutter August 1, 2002
- "Smart Pointers - What, Why, Which?" by Yonat Sharon
- "Smart Pointers Overview" by John M. Dlugosz
- The YASPER library Yet Another Smart Pointer implementation in C++
- Smart Pointers in Delphi
- Smart Pointers in Rust