C++ Client API Reference for Oracle Coherence
14c (14.1.2.0.0)

F79659-03

coherence/lang/Immutable.hpp

00001 /*
00002  * Copyright (c) 2000, 2020, Oracle and/or its affiliates.
00003  *
00004  * Licensed under the Universal Permissive License v 1.0 as shown at
00005  * http://oss.oracle.com/licenses/upl.
00006  */
00007 #ifndef COH_IMMUTABLE_HPP
00008 #define COH_IMMUTABLE_HPP
00009 
00010 #include "coherence/lang/compatibility.hpp"
00011 
00012 #include "coherence/lang/IllegalArgumentException.hpp"
00013 #include "coherence/lang/Object.hpp"
00014 #include "coherence/lang/TypedHandle.hpp"
00015 
00016 #include <sstream>
00017 
00018 COH_OPEN_NAMESPACE2(coherence,lang)
00019 
00020 
00021 /**
00022 * A Handle implementation which upon assignment ensures that it will
00023 * reference an immutable Object. If the passed Object is immutable
00024 * (as defined by Object::isImmutable()) then the handle simply holds a const
00025 * pointer to the Object. If the passed Object is not immutable, then a clone
00026 * attempt is made, and held via a const pointer.  If the supplied Object is
00027 * not immutable and not cloneable then assignment will fail with a
00028 * CloneNotSupport exception.
00029 *
00030 * @author mf  2007.07.05
00031 */
00032 template<class T>
00033 class Immutable
00034     : public TypedHandle<const T>
00035     {
00036     // ----- constructors ---------------------------------------------------
00037 
00038     public:
00039         /**
00040         * Construct a @c NULL handle.
00041         */
00042         Immutable()
00043             : TypedHandle<const T>()
00044             {
00045             }
00046 
00047         /**
00048         * Construct an Immutable handle to the given Object.
00049         *
00050         * @param po the pointer to the Object to reference
00051         */
00052         Immutable(const T* po)
00053             : TypedHandle<const T>(NULL)
00054             {
00055             set(po, NULL);
00056             }
00057 
00058         /**
00059         * Construct an Immutable handle from another Handle
00060         *
00061         * @param that another handle
00062         */
00063         template<class O> Immutable<T>(const TypedHandle<O>& that)
00064             : TypedHandle<const T>(that)
00065             {
00066             }
00067 
00068         /**
00069         * Construct an Immutable handle from another Handle
00070         *
00071         * @param that  another handle
00072         */
00073         Immutable<T>(const Immutable<T>& that)
00074             : TypedHandle<const T>(that)
00075             {
00076             }
00077 
00078     // ----- helper methods -------------------------------------------------
00079 
00080     protected:
00081         /**
00082         * Set this handle to point to immutable version of the given object,
00083         * or unset the handle.
00084         *
00085         * If the given pointer points to the immutable object, then this
00086         * handle will point it. If the given pointer points to the mutable
00087         * object, then this handle will point to the clone of that object.
00088         * If the given pointer is NULL, then this handle will point to no
00089         * object.
00090         *
00091         * @param cpo    pointer to the target object
00092         * @param pThat  pointer to the source ChainedHandleElement
00093         */
00094         Immutable<T>& set(const T* cpo, const ChainedHandleElement* pThat)
00095             {
00096             if (cpo == NULL || cpo->isImmutable())
00097                 {
00098                 if (pThat != NULL)
00099                     {
00100                     TypedHandle<const T>::set(cpo, *pThat);
00101                     }
00102                 else
00103                     {
00104                     TypedHandle<const T>::operator=(cpo);
00105                     }
00106                 }
00107             else
00108                 {
00109                 Object::View vClone;
00110                     {
00111                     // With the Sun compiler the temporary handle created
00112                     // here will live for the duration of the surrounding
00113                     // scope, we must pop the scope to ensure that the
00114                     // temporary handle is destroyed/detached, so the
00115                     // immutability check can pass
00116                     vClone = cpo->clone();
00117                     }
00118                 if (NULL == vClone || !vClone->isImmutable())
00119                     {
00120                     COH_THROW_STREAM(IllegalArgumentException,
00121                         "Object of type '" << typeid(*cpo).name() <<
00122                         "' cannot be made immutable through cloning.");
00123                     }
00124                 TypedHandle<const T>::operator=(cast<typename T::View>(vClone));
00125                 }
00126             return *this;
00127             }
00128     };
00129 
00130 /**
00131 * Return an immutable version of the supplied Object. If the supplied view
00132 * does not refer to an immutable Object, then a clone of the original Object
00133 * is returned.
00134 *
00135 * @param v  view to the Object to return an immutable copy of
00136 *
00137 * @return a view to an immutable version of the supplied Object, or the
00138 *         supplied Object if it was already immutable
00139 */
00140 template<class T> typename T::View immutable_view(const typename T::View& v)
00141     {
00142     return Immutable<T>(v);
00143     }
00144 
00145 COH_CLOSE_NAMESPACE2
00146 
00147 #endif // COH_IMMUTABLE_HPP
Copyright © 2000, 2025, Oracle and/or its affiliates. Licensed under the Universal Permissive License v 1.0 as shown at https://oss.oracle.com/licenses/upl.