1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 | #ifndef STLPLUS_SMART_PTR #define STLPLUS_SMART_PTR //////////////////////////////////////////////////////////////////////////////// // Author: Andy Rushton // Copyright: (c) Southampton University 1999-2004 // (c) Andy Rushton 2004 onwards // License: BSD License, see ../docs/license.html // A smart pointer is a memory-managing pointer to an object. If you like, it // is a zero-dimensional container. // Assignment of smart pointers result in multiple aliases of the same object. // The term alias is used to differentiate from conventional pointers because // the semantics are different. // Aliases can be turned into copies if the pointed-to class supports copying. // The base class is smart_ptr_base which defines the common interface. Then // there are three subclasses which have the same interface but different copy // semantics: // - smart_ptr for simple types and classes which have copy constructors // - smart_ptr_clone for polymorphic class hierarchies which are copied using a clone method // - smart_ptr_nocopy for any class that cannot or should not be copied //////////////////////////////////////////////////////////////////////////////// #include "containers_fixes.hpp" #include "exceptions.hpp" #include "copy_functors.hpp" #include <map> #include <string> namespace stlplus { //////////////////////////////////////////////////////////////////////////////// // internals template < typename T> class smart_ptr_holder; //////////////////////////////////////////////////////////////////////////////// // Base class //////////////////////////////////////////////////////////////////////////////// template < typename T, typename C> class smart_ptr_base { public : ////////////////////////////////////////////////////////////////////////////// // member type definitions typedef T value_type; typedef T& reference; typedef const T& const_reference; typedef C value_copy; ////////////////////////////////////////////////////////////////////////////// // constructors and destructors // create a null pointer smart_ptr_base ( void ); // create a pointer containing a dynamically created object // Note: the object must be allocated *by the user* with new // constructor form - must be called in the form smart_ptr_base<type> x(new type(args)) explicit smart_ptr_base (T* data); // copy constructor implements aliasing so no copy is made // note that the copy constructor should NOT be explicit, as this breaks // the returning of pointer objects from functions (at least within GCC 4.4) smart_ptr_base ( const smart_ptr_base <T,C>& r); // assignment operator - required, else the output of GCC suffers segmentation faults smart_ptr_base <T,C>& operator =( const smart_ptr_base <T,C>& r); // destructor decrements the reference count and delete only when the last reference is destroyed ~ smart_ptr_base ( void ); ////////////////////////////////////////////////////////////////////////////// // logical tests to see if there is anything contained in the pointer since it can be null // there are two forms:explicit and implicit // implicit: if(!r) or if(r) // explicit: if(r.null()) or if(r.present()) operator bool ( void ) const ; bool operator !( void ) const ; bool present( void ) const ; bool null( void ) const ; ////////////////////////////////////////////////////////////////////////////// // dereference operators and functions // dereference the smart pointer to get the object - use in the form *p1 // exceptions: null_dereference T& operator *( void ) ; // exceptions: null_dereference const T& operator *( void ) const ; // used as a prefix to a member access to the contained object e.g. p1->print() calls T::print() // exceptions: null_dereference T* operator ->( void ) ; // exceptions: null_dereference const T* operator ->( void ) const ; ////////////////////////////////////////////////////////////////////////////// // explicit function forms of the above assignment and dereference operators // get the value // exceptions: null_dereference T& value( void ) ; // exceptions: null_dereference const T& value( void ) const ; // set the pointer // deletes the previous pointer and adopts the passed pointer instead // Note: the object must be allocated *by the user* with new // Warning: it is very easy to break the memory management with this operation void set (T* data = 0); // get the pointer T* pointer( void ); const T* pointer( void ) const ; ////////////////////////////////////////////////////////////////////////////// // functions to manage aliases // make this an alias of the passed object void alias( const smart_ptr_base <T,C>&); // test whether two pointers point to the same object(known as aliasing the object) // used in the form if(a.aliases(b)) bool aliases( const smart_ptr_base <T,C>&) const ; // find the number of aliases - used when you need to know whether an // object is still referred to from elsewhere (rare!) unsigned alias_count( void ) const ; // delete the object and make the pointer null - does not make it unique // first, so all other pointers to this will be null too void clear( void ); // make the pointer unique and null in one step - does not affect other // pointers that were pointing to the same object void clear_unique( void ); ////////////////////////////////////////////////////////////////////////////// // functions that involve copying // these functions use the copy functor passed as the template parameter C // to copy the object with the right copy semantics. If the copy functor // is no_copy, an exception will be thrown. // create a pointer containing a *copy* of the object using the template parameter C // this copy is taken because the pointer class maintains a dynamically allocated object // and the T& may not be (usually is not) dynamically allocated // exceptions: illegal_copy explicit smart_ptr_base ( const T& data) ; // set the value - note that this does a copy using the C template parameter // exceptions: illegal_copy void set_value( const T& data) ; // make this pointer unique with respect to any other references to the same object // if this pointer is already unique, it does nothing - otherwise it copies the object // exceptions: illegal_copy void make_unique( void ) ; // make this pointer a unique copy of the parameter // useful for expressions like p1.copy(p2) which makes p1 a pointer to a unique copy of the contents of p2 // exceptions: illegal_copy void copy( const smart_ptr_base <T,C>&) ; protected : smart_ptr_holder<T>* m_holder; public : // internal use only - had to make them public because they need to be // accessed by routines that could not be made friends smart_ptr_holder<T>* _handle( void ) const ; void _make_alias(smart_ptr_holder<T>* handle); }; //////////////////////////////////////////////////////////////////////////////// // smart_ptr for simple types and classes which have copy constructors template < typename T> class smart_ptr : public smart_ptr_base <T, constructor_copy <T> > { public : smart_ptr ( void ) {} explicit smart_ptr ( const T& data) : smart_ptr_base <T, constructor_copy <T> >(data) {} explicit smart_ptr (T* data) : smart_ptr_base <T, constructor_copy <T> >(data) {} smart_ptr <T>& operator =( const T& data) { this ->set_value(data); return * this ;} smart_ptr <T>& operator =(T* data) { this -> set (data); return * this ;} ~ smart_ptr ( void ) {} }; //////////////////////////////////////////////////////////////////////////////// // smart_ptr_clone for polymorphic class hierarchies which have a clone method template < typename T> class smart_ptr_clone : public smart_ptr_base <T, clone_copy <T> > { public : smart_ptr_clone ( void ) {} explicit smart_ptr_clone ( const T& data) : smart_ptr_base <T, clone_copy <T> >(data) {} explicit smart_ptr_clone (T* data) : smart_ptr_base <T, clone_copy <T> >(data) {} smart_ptr_clone <T>& operator =( const T& data) { this ->set_value(data); return * this ;} smart_ptr_clone <T>& operator =(T* data) { this -> set (data); return * this ;} ~ smart_ptr_clone ( void ) {} }; //////////////////////////////////////////////////////////////////////////////// // smart_ptr_nocopy for any class that cannot or should not be copied template < typename T> class smart_ptr_nocopy : public smart_ptr_base <T, no_copy <T> > { public : smart_ptr_nocopy ( void ) {} explicit smart_ptr_nocopy (T* data) : smart_ptr_base <T, no_copy <T> >(data) {} smart_ptr_nocopy <T>& operator =(T* data) { this -> set (data); return * this ;} ~ smart_ptr_nocopy ( void ) {} }; //////////////////////////////////////////////////////////////////////////////// } // end namespace stlplus #include "smart_ptr.tpp" #endif |