The documentation is divided into a an explanation for each container. All the common interface is explained only once, but links are always provided to the relevant links. Please make sure you understand the Clonable concept and the Clone Allocator concept.
Refinement of
The Clonable concept is introduced to formalize the requirements for copying heap-allocated objects. A type T might be Clonable even though it is not Assignable or Copy Constructible. Notice that many operations on the containers does not even require the stored type to be Clonable.
Notation
Type | Object (const or non-const) | Pointer | Describes |
T | a | ptr | A Clonable type |
Valid expressions
Expression | Type | Semantics |
new_clone(a); | T* | Allocate a new object that can be considered equivalent to the a object |
delete_clone(ptr); | void | Deallocate an object previously allocated with allocate_clone(). Must not throw |
In the <boost/ptr_container/clone_allocator.hpp> header a default implementation of the two functions is given:
namespace boost { template< class T > inline T* new_clone( const T& t ) { return new T( t ); } template< class T > void delete_clone( const T* t ) { checked_delete( r ); } }
Notice that this implementation makes normal Copy Constructible classes are automatically Clonable unless operator new() or operator delete() are hidden.
The two functions represent a layer of indirection which is necessary to support classes that are not Copy Constructible by default. Notice that the implementation relies on argument-dependent lookup (ADL) to find the right version of new_clone() and delete_clone(). This means that one does not need to overload or specialize the function is the boost namespace, but it can be placed together with the rest of the interface of the class. If you are implementing a class inline in headers, remember to forward declare the functions.
The Clone Allocator concept is introduced to formalize the way pointer containers controls memory of the stored objects (and not the pointers to the stored objects). The clone allocator allows users to apply custom allocators/deallocators for the cloned objects.
More information can be found below:
Notation
Type | Object (const or non-const) | Describes |
T | a | A type |
T* | ptr | A pointer to T |
Valid expressions
Expression | Type | Semantics |
CloneAllocator::allocate_clone(a); | T* | Allocate a new object that can be considered equivalent to the a object |
CloneAllocator::deallocate_clone(ptr); | void | Deallocate an object previously allocated with CloneAllocator::allocate_clone() or a compatible allocator. Must not throw. |
The library comes with two predefined clone allocators.
This is the default clone allocator used by all pointer containers. For most purposes you will never have to change this default.
Definition
namespace boost { struct heap_clone_allocator { template< class U > static U* allocate_clone( const U& r ) { return new_clone( r ); } template< class U > static void deallocate_clone( const U* r ) const { delete_clone( r ); } }; }
Notice that the above definition allows you to support custom allocation schemes by relying on new_clone() and delete_clone().
This class provides a way to remove ownership properties of the pointer containers. As its name implies, this means that you can instead use the pointer containers as a view into an existing container.
Definition
namespace boost { struct view_clone_allocator { template< class U > static U* allocate_clone( const U& r ) { return const_cast<U*>(&r); } template< class U > static void deallocate_clone( const U* ) { // empty } }; }
See also
The pointer container adapters are used when you want to make a pointer container starting from your own "normal" container. For example, you might have a map class that is extends std::map in some way; the adapter class then allows you to use your map class as a basis for a new pointer container.
The library provides an adapter for each type of standard container:
The pointer containers of this library are all built using the pointer container adapters. There is a pointer container for each type of "normal" standard container:
The map iterators are a bit different compared to the normal ones. The reason is that it is a bit clumsy to access the key and the mapped object through i->first and i->second, and one tends to forget what is what. Moreover, and more importantly, we also want to hide the pointer as much as possibble. The new style can be illustrated with a small example:
typedef ptr_map<string,int> map_t; map_t m; m[ "foo" ] = 4; // insert pair m[ "bar" ] = 5; // ditto ... for( map_t::iterator i = m.begin(); i != m.end(); ++i ) { *i += 42; // add 42 to each value cout << "value=" << *i << ", key=" << i.key() << "n"; }
So the difference from the normal map iterator is that
The purpose of the class is simply to tell the containers that null values should be allowed. Its definition is trivial:
namespace boost { template< class T > struct nullable { typedef T type; }; }
Please notice that nullable has no effect on the containers interface (except for is_null() functions). For example, it does not make sense to do
boost::ptr_vector< boost::nullable<T> > vec; vec.push_back( new boost::nullable<T> ); // no no boost::nullable<T>& ref = vec[0]; // also no no
There are three exceptions that are thrown by this library. The exception hierarchy looks as follows:
namespace boost { class bad_ptr_container_operation : public std::exception { public: bad_ptr_container_operation( const char* what ); }; class bad_index : public bad_ptr_container_operation { public: bad_index( const char* what ); }; class bad_pointer : public bad_ptr_container_operation { public: bad_pointer(); bad_pointer( const char* what ); }; }
copyright: | Thorsten Ottosen 2004-2005. |
---|