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

F79659-03

coherence/lang/WeakHolder.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_WEAK_HOLDER_HPP
00008 #define COH_WEAK_HOLDER_HPP
00009 
00010 #include "coherence/lang/compatibility.hpp"
00011 
00012 #include "coherence/lang/Object.hpp"
00013 #include "coherence/lang/TypedHandle.hpp"
00014 #include "coherence/lang/TypedHolder.hpp"
00015 #include "coherence/lang/MemberHolder.hpp"
00016 #include "coherence/lang/WeakReference.hpp"
00017 
00018 #include <ostream>
00019 
00020 COH_OPEN_NAMESPACE2(coherence,lang)
00021 
00022 
00023 /**
00024 * WeakHolders are a TypedHolder like wrapper around WeakReferences.
00025 * WeakReferences allow one Object to safely reference another without blocking
00026 * it from being destroyed. WeakRefereces are necessary when building Object
00027 * graphs to avoid leaking the Object graph due to cyclical references.
00028 *
00029 * WeakReferences are automatically NULL'd out when their referenced Object
00030 * is destroyed. Additionally if the weakly referenced Object becomes only
00031 * referenced via Views, then the WeakReference and corresponding WeakHandle
00032 * will only return View to the Object, if an attempt is made to obtain a
00033 * Handle a NULL will be returned instead.
00034 *
00035 * WeakHolders are wrappers for MemberHolder, and provide a thread-safe handle
00036 * implementation.  As such they require references to both the referenced
00037 * "child" Object, as well as the owning "parent" Object. As with MemberHolder
00038 * the parent object must inherit from Object.
00039 *
00040 * Note: In the rare case that a WeakHolder is declared via the mutable
00041 *       keyword, the WeakHolder must be informed of this fact by setting
00042 *       fMutable to true during construction.
00043 *
00044 * @see MemberHolder
00045 *
00046 * @author mf  2008.12.09
00047 */
00048 template<class T>
00049 class WeakHolder
00050     {
00051     // ----- typedefs -------------------------------------------------------
00052 
00053     public:
00054         /**
00055         * The type of Object which this handle weakly references.
00056         */
00057         typedef const T ValueType;
00058 
00059         /**
00060         * The associated non-weak Handle type.
00061         */
00062         typedef typename T::Handle ValueHandle;
00063 
00064         /**
00065         * The associated non-weak View type.
00066         */
00067         typedef typename T::View ValueView;
00068 
00069         /**
00070         * The associated non-weak Holder type.
00071         */
00072         typedef typename T::Holder ValueHolder;
00073 
00074 
00075     // ----- constructors ---------------------------------------------------
00076 
00077     public:
00078         /**
00079         * Construct a WeakHolder referencing NULL.
00080         *
00081         * @param oGuardian  reference to the WeakHolder's parent
00082         */
00083         WeakHolder(const Object& oGuardian)
00084             : m_ohWeak(oGuardian, WeakReference::valueOf(NULL)),
00085               m_cpWeak(NULL)
00086             {
00087             }
00088 
00089         /**
00090         * Construct a WeakHolder referencing an object.
00091         *
00092         * @param oGuardian  reference to the WeakHolder's parent
00093         * @param that       the object to reference
00094         */
00095         WeakHolder(const Object& oGuardian, const TypedHolder<T>& that)
00096             : m_ohWeak(oGuardian, WeakReference::valueOf(that)),
00097               m_cpWeak(get_pointer(that))
00098             {
00099             }
00100 
00101         /**
00102         * Construct a WeakHolder referencing an object.
00103         *
00104         * @param oGuardian  reference to the WeakHolder's parent
00105         * @param that       the object to reference
00106         * @param fMutable   true if the member is declared as mutable, false
00107         *                   if declared as const
00108         */
00109         WeakHolder(const Object& oGuardian, const TypedHolder<T>& that,
00110                 bool fMutable)
00111             : m_ohWeak(oGuardian, WeakReference::valueOf(that), fMutable),
00112               m_cpWeak(get_pointer(that))
00113             {
00114             }
00115 
00116 
00117     // ----- operators ------------------------------------------------------
00118 
00119     public:
00120         /**
00121         * Re-assign the WeakHolder to weakly reference a new Object.
00122         *
00123         * @param that  the object to reference
00124         */
00125         WeakHolder& operator=(const TypedHolder<T>& that)
00126             {
00127             WeakReference::Holder ohRef  = WeakReference::valueOf(that);
00128             const T*              cpThat = get_pointer(that);
00129 
00130             // update m_ohWeak and m_cpWeak under a single write lock
00131                 {
00132                 SynchronizedMemberWriteBlock guard(get_guardian(m_ohWeak));
00133                 guard.setMember(m_ohWeak, ohRef);
00134                 m_cpWeak = cpThat;
00135                 }
00136 
00137             return *this;
00138             }
00139 
00140         /**
00141         * Re-assign the WeakHolder to weakly reference a new Object.
00142         *
00143         * @param that  the object to reference
00144         */
00145         WeakHolder& operator=(const WeakHolder& that)
00146             {
00147             return operator=((ValueHolder) that);
00148             }
00149 
00150         /**
00151         * Return a non-weak View to the weakly referenced Object, or a
00152         * View to NULL if the weakly referenced Object has been
00153         * destroyed.
00154         *
00155         * @return ValueView
00156         */
00157         operator ValueView() const
00158             {
00159             WeakReference::View vRef;
00160             const T*            cpWeak;
00161 
00162             // read m_ohWeak and m_cpWeak under a single read lock
00163                 {
00164                 SynchronizedMemberReadBlock guard(get_guardian(m_ohWeak));
00165                 vRef   = guard.getMember(m_ohWeak);
00166                 cpWeak = m_cpWeak;
00167                 }
00168 
00169             Object::View v = vRef->get();
00170             return NULL == v
00171                 ? TypedHandle<const T>() // Object has been collected
00172                 : TypedHandle<const T>(cpWeak, v);
00173             }
00174 
00175         /**
00176         * Return a View to the referenced Object.
00177         *
00178         * @return a View to the referenced Object
00179         */
00180         template<class PT>
00181         operator TypedHandle<const PT>() const
00182             {
00183             return (ValueView) *this;
00184             }
00185 
00186         /**
00187         * Return a TypedHolder to the referenced Object.
00188         *
00189         * @return a TypedHolder to the referenced Object
00190         */
00191         template<class PT>
00192         operator TypedHolder<PT>() const
00193             {
00194             WeakReference::Holder ohRef;
00195             const T*              cpWeak;
00196 
00197             // read m_hWeak and m_pWeak under a single read lock
00198                 {
00199                 SynchronizedMemberReadBlock guard(get_guardian(m_ohWeak));
00200                 ohRef  = guard.getMember(m_ohWeak);
00201                 cpWeak = m_cpWeak;
00202                 }
00203 
00204             WeakReference::Handle hRef = cast<WeakReference::Handle>(ohRef,
00205                     /*fThrow*/ false);
00206             Object::Holder        oh   = NULL == hRef
00207                 ? ohRef->get()
00208                 : hRef->get();
00209 
00210             if (NULL == oh)
00211                 {
00212                 return NULL; // Object has been collected
00213                 }
00214             else if (NULL == hRef)
00215                 {
00216                 return TypedHandle<const T>(cpWeak, oh); // View
00217                 }
00218             else
00219                 {
00220                 return TypedHandle<T>(const_cast<T*>(cpWeak), oh); // Handle
00221                 }
00222             }
00223 
00224         /**
00225         * Dereference the WeakHolder returning a non-weak View.
00226         *
00227         * If the non-weak View is non-NULL then it is guaranteed
00228         * that the Object will continue to exist for at least the
00229         * life of the returned View.
00230         *
00231         * @return a non-weak View to the referenced Object
00232         */
00233         ValueView operator->() const
00234             {
00235             return (ValueView) *this;
00236             }
00237 
00238         /**
00239         * Dereference this handle, returning <tt>T&</tt>.
00240         *
00241         * @return a raw <tt>T&</tt> reference to the referenced Object
00242         *
00243         * @throws NullPointerException if the this handle is @c NULL
00244         */
00245         const T& operator*() const
00246             {
00247             return *operator->();
00248             }
00249 
00250     // ----- data members ---------------------------------------------------
00251 
00252     protected:
00253         /**
00254         * The Holder to the associated WeakReference.
00255         */
00256         MemberHolder<WeakReference> m_ohWeak;
00257 
00258         /**
00259         * Raw pointer to the weakly referenced object.
00260         */
00261         const T* m_cpWeak;
00262     };
00263 
00264 
00265 // ----- non-member operators and functions ---------------------------------
00266 
00267 /**
00268 * Output a human-readable description of the given
00269 * WeakHolder<T> to the specified stream.
00270 *
00271 * @param out  the stream used to output the description
00272 * @param woh   the WeakHolder<T> to describe
00273 *
00274 * @return the supplied stream
00275 */
00276 template <typename Char, typename Traits, class T>
00277 COH_INLINE std::basic_ostream<Char, Traits>& operator<<(std::basic_ostream<Char, Traits>& out, const WeakHolder<T>& woh)
00278     {
00279     typename WeakHolder<T>::ValueView v = woh;
00280     out << v;
00281     return out;
00282     }
00283 
00284 /**
00285 * Assign the specified handle to NULL.
00286 *
00287 * @param woh the handle to clear
00288 */
00289 template<class T> void clear_handle(WeakHolder<T>& woh)
00290     {
00291     woh = NULL;
00292     }
00293 
00294 /**
00295 * Return true if the supplied handle equals NULL.
00296 *
00297 * @param woh  the weak handle to test
00298 *
00299 * @return true iff the supplied handle equals NULL
00300 */
00301 template<class T>
00302 bool is_null(const WeakHolder<T>& woh)
00303     {
00304     return woh == NULL;
00305     }
00306 
00307 /**
00308 * Perform a dynamic cast the pointer associated with the WeakHolder
00309 * to a the specified handle/view type.
00310 *
00311 * @param woh      the WeakHolder from which to perform the cast
00312 * @param fThrow   true if an exception is to be thrown on a failed cast
00313 *
00314 * @return the casted pointer, or NULL if the cast fails and fThrow is false
00315 *
00316 * @throws ClassCastException if the cast fails and fThrow is true
00317 */
00318 template<class D, class T>
00319 D cast(const WeakHolder<T>& woh, bool fThrow = true)
00320     {
00321     return cast<D>((TypedHolder<T>) woh, fThrow);
00322     }
00323 
00324 /**
00325 * Perform an instanceof check on a handle or view.
00326 *
00327 * @param woh  the WeakHolder from which to perform the test
00328 *
00329 * @return true if the supplied handle is an instance of the specified type
00330 */
00331 template<class D, class T>
00332 bool instanceof(const WeakHolder<T>& woh)
00333     {
00334     return NULL != cast<D>(woh, false);
00335     }
00336 
00337 COH_CLOSE_NAMESPACE2
00338 
00339 #endif // COH_WEAK_HOLDER_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.