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

F79659-03

coherence/lang/MemberView.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_MEMBER_VIEW_HPP
00008 #define COH_MEMBER_VIEW_HPP
00009 
00010 #include "coherence/lang/compatibility.hpp"
00011 
00012 #include "coherence/lang/Object.hpp"
00013 #include "coherence/lang/SmartMember.hpp"
00014 #include "coherence/lang/SynchronizedMemberReadBlock.hpp"
00015 #include "coherence/lang/SynchronizedMemberWriteBlock.hpp"
00016 #include "coherence/lang/TypedHandle.hpp"
00017 #include "coherence/lang/TypedHolder.hpp"
00018 
00019 #include <ostream>
00020 
00021 COH_OPEN_NAMESPACE2(coherence,lang)
00022 
00023 /**
00024 * MemberView is a thread-safe view intended for use as a data-member within
00025 * Objects.
00026 *
00027 * Note: In the rare case that a MemberView is declared via the mutable
00028 *       keyword, the MemberView must be informed of this fact by setting
00029 *       fMutable to true during construction.
00030 *
00031 * @author mf  2008.01.09
00032 *
00033 * @see MemberHandle
00034 * @see MemberHolder
00035 */
00036 template<class T>
00037 class MemberView
00038     : public SmartMember, public ChainedHandleElement
00039     {
00040     // ----- typedefs -------------------------------------------------------
00041 
00042     public:
00043         /**
00044         * The type of the values the holder can reference.
00045         */
00046         typedef T ValueType;
00047 
00048         /**
00049         * The View type for the referenced Object.
00050         */
00051         typedef typename T::View ValueView;
00052 
00053         /**
00054         * Result type for a non-const get operation.
00055         */
00056         typedef ValueView GetType;
00057 
00058 
00059     // -------- constructors ------------------------------------------------
00060 
00061     public:
00062         /**
00063         * Construct a new MemberView referencing NULL via a handle.
00064         *
00065         * @param oGuardian  the object that protects this member
00066         */
00067         MemberView(const Object& oGuardian)
00068             : SmartMember(oGuardian), ChainedHandleElement(/*fView*/ true),
00069               m_cpo(NULL)
00070             {
00071             if (oGuardian._isEscaped())
00072                 {
00073                 m_prev = m_next = NULL;
00074                 }
00075             }
00076 
00077         /**
00078         * Construct a new MemberView referencing specified Object.
00079         *
00080         * @param oGuardian  the object that protects this member
00081         * @param that       the object to reference
00082         */
00083         MemberView(const Object& oGuardian, const ValueView& that)
00084             : SmartMember(oGuardian),  ChainedHandleElement(/*fView*/ true),
00085               m_cpo(NULL)
00086             {
00087             if (oGuardian._isEscaped())
00088                 {
00089                 m_prev = m_next = NULL;
00090                 }
00091             set(that);
00092             }
00093 
00094         /**
00095         * Construct a new MemberView referencing specified Object.
00096         *
00097         * @param oGuardian  the object that protects this member
00098         * @param that       the object to reference
00099         * @param fMutable   true if the object was declared as mutable
00100         */
00101         MemberView(const Object& oGuardian, const ValueView& that, bool fMutable)
00102             : SmartMember(oGuardian), ChainedHandleElement(/*fView*/ true),
00103               m_cpo(NULL)
00104             {
00105             if (oGuardian._isEscaped())
00106                 {
00107                 m_prev = m_next = NULL;
00108                 }
00109             set(that);
00110             m_nMutability = fMutable ? forever_mutable : safe_immutable;
00111             }
00112 
00113         /**
00114         * Destroy the MemberView.
00115         */
00116         ~MemberView()
00117             {
00118             try
00119                 {
00120                 m_nMutability = inherited;
00121                 set(NULL);
00122                 }
00123             catch (const std::exception& e)
00124                 {
00125                 // Exception::View is not a known type within this file
00126                 std::cerr << "Error during ~MemberView: " << e.what() << std::endl;
00127                 return; // can't re-throw from within destructor
00128                 }
00129             }
00130 
00131     private:
00132         /**
00133         * Blocked copy constructor.
00134         */
00135         MemberView(const MemberView&);
00136 
00137 
00138     // ----- operators ------------------------------------------------------
00139 
00140     public:
00141         /**
00142         * Assign the MemberView to reference another object.
00143         *
00144         * @param that  the object to reference
00145         *
00146         * @return a reference to this MemberView
00147         */
00148         MemberView& operator=(const ValueView& that)
00149             {
00150             set(that);
00151             return *this;
00152             }
00153 
00154         /**
00155         * Assign the MemberView to reference another object.
00156         *
00157         * @param that  the object to reference
00158         *
00159         * @return a reference to this MemberView
00160         */
00161         MemberView& operator=(const MemberView& that)
00162             {
00163             set(that);
00164             return *this;
00165             }
00166 
00167         /**
00168         * Return a View to the referenced Object.
00169         *
00170         * @return a View to the referenced Object
00171         */
00172         operator ValueView() const
00173             {
00174             return get();
00175             }
00176 
00177         /**
00178         * Return a View to the referenced Object.
00179         *
00180         * @return a View to the referenced Object
00181         */
00182         template<class PT>
00183         operator TypedHandle<const PT>() const
00184             {
00185             return get();
00186             }
00187 
00188         /**
00189         * Return a TypedHolder to the referenced Object.
00190         *
00191         * @return a TypedHolder to the referenced Object
00192         */
00193         template<class PT>
00194         operator TypedHolder<PT>() const
00195             {
00196             return get();
00197             }
00198 
00199         /**
00200         * Dereference the MemberView.
00201         *
00202         * @return a const pointer to the referenced Object
00203         */
00204         ValueView operator->() const
00205             {
00206             return get();
00207             }
00208 
00209         /**
00210         * Dereference this handle, returning <tt>T&</tt>.
00211         *
00212         * @return a raw <tt>T&</tt> reference to the referenced Object
00213         *
00214         * @throws NullPointerException if the this handle is @c NULL
00215         */
00216         const T& operator*() const
00217             {
00218             return *get();
00219             }
00220 
00221     // ----- SmartMember interface ------------------------------------------
00222 
00223     protected:
00224         /**
00225         * {@inheritDoc}
00226         */
00227         virtual void onEscape(bool fEscaped) const
00228             {
00229             const T* cpo = m_cpo;
00230             if (cpo)
00231                 {
00232                 cpo->_attach(fEscaped);
00233                 }
00234 
00235             if (fEscaped)
00236                 {
00237                 performAction(unlink());
00238                 m_prev = m_next = NULL;
00239                 }
00240             else
00241                 {
00242                 if (cpo)
00243                     {
00244                     cpo->_detach(/*fEscaped*/ true);
00245                     }
00246                 m_prev = m_next = this;
00247                 }
00248 
00249             SmartMember::onEscape(fEscaped);
00250             }
00251 
00252         /**
00253         * {@inheritDoc}
00254         */
00255         virtual size64_t retained() const
00256             {
00257             ValueView v = get();
00258             return v == NULL
00259                     ? 0
00260                     : v->sizeOf(/*fDeep*/ true);
00261             }
00262 
00263 
00264     // ----- helper methods -------------------------------------------------
00265 
00266     protected:
00267         /**
00268         * Set the view to reference an Object via a View.
00269         *
00270         * @param that   the Object to reference
00271         * @param pSync  optional external SyncornizedMemberWriteBlock to use
00272         *               to avoid internal synchronization.
00273         */
00274         void set(const TypedHolder<T>& that,
00275                  SynchronizedMemberWriteBlock* pSync = NULL)
00276             {
00277             if (m_prev == NULL)
00278                 {
00279                 setEscaped(that, pSync);
00280                 }
00281             else if (m_nMutability >= forever_immutable)
00282                 {
00283                 coh_throw_illegal_state("attempt to set const MemberView");
00284                 }
00285             else
00286                 {
00287                 performAction(link(that));
00288                 m_cpo = get_pointer(that);
00289                 }
00290             }
00291 
00292         /**
00293         * Set the escaped view to reference an Object via a View.
00294         *
00295         * @param that   the Object to reference
00296         * @param pSync  optional external SyncornizedMemberWriteBlock to use
00297         *               to avoid internal synchronization.
00298         */
00299         void setEscaped(const TypedHolder<T>& that,
00300                  SynchronizedMemberWriteBlock* pSync = NULL)
00301             {
00302             const Object& oGuardian = getGuardian();
00303             const T*      cpo       = get_pointer(that);
00304             const T*      cpDetach  = NULL;
00305             const Object* cpAttach  = NULL == cpo ? NULL : cpo->_attach(/*fEscaped*/ true);
00306 
00307             if (pSync == NULL)
00308                 {
00309                 // sync block
00310                 SynchronizedMemberWriteBlock::Guard guard(oGuardian);
00311                 if (m_nMutability >= forever_immutable)
00312                     {
00313                     coh_throw_illegal_state("attempt to set const MemberView");
00314                     }
00315 
00316                 cpDetach = m_cpo;
00317                 m_cpo    = NULL == cpAttach ? NULL : cpo;
00318                 }
00319             else
00320                 {
00321                 // sync block
00322                 SynchronizedMemberWriteBlock syncWrite(oGuardian, pSync);
00323                 if (m_nMutability >= forever_immutable)
00324                     {
00325                     coh_throw_illegal_state("attempt to set const MemberView");
00326                     }
00327 
00328                 cpDetach = m_cpo;
00329                 m_cpo    = NULL == cpAttach ? NULL : cpo;
00330                 }
00331 
00332             if (cpDetach)
00333                 {
00334                 cpDetach->_detach(/*fEscaped*/ true);
00335                 }
00336 
00337             }
00338 
00339         /**
00340         * Return a View to the referenced Object.
00341         *
00342         * @param pSync  optional external SyncornizedMemberReadBlock to use
00343         *               to avoid internal synchronization.
00344         *
00345         * @return a View to the referenced Object
00346         */
00347         ValueView get(SynchronizedMemberReadBlock* pSync = NULL) const
00348             {
00349             if (m_prev == NULL)
00350                 {
00351                 return getEscaped(pSync);
00352                 }
00353             else
00354                 {
00355                 return TypedHandle<const T>(m_cpo, *this);
00356                 }
00357             }
00358 
00359         /**
00360         * Return a View to the referenced Object.
00361         *
00362         * @param pSync  optional external SyncornizedMemberReadBlock to use
00363         *               to avoid internal synchronization.
00364         *
00365         * @return a View to the referenced Object
00366         */
00367         ValueView getEscaped(SynchronizedMemberReadBlock* pSync = NULL) const
00368             {
00369             const Object& oGuardian = getGuardian();
00370             if (pSync != NULL)
00371                 {
00372                 // sync block
00373                 SynchronizedMemberReadBlock syncRead(oGuardian, pSync);
00374                 return ValueView(m_cpo); // must occur within scope of syncRead
00375                 }
00376             else if (m_nMutability == safe_immutable)
00377                 {
00378                 // safe_immutable, synchronization not required
00379                 return ValueView(m_cpo);
00380                 }
00381             else
00382                 {
00383                 // sync block
00384                 SynchronizedMemberReadBlock::Guard guard(oGuardian);
00385                 return ValueView(m_cpo); // must occur within scope of guard
00386                 }
00387             }
00388 
00389 
00390 
00391         /**
00392         * Perform the specified action.
00393         *
00394         * @param nAction  the action to perform
00395         */
00396         void performAction(Action nAction) const
00397             {
00398             const T* cpo = m_cpo;
00399             switch (nAction)
00400                 {
00401                 case action_error:
00402                 case action_flip:
00403                     coh_throw_illegal_state("corrupted ChainedHandleElement");
00404                 case action_detach:
00405                     if (cpo) cpo->_detach(/*fEscaped*/ false);
00406                     // fall through
00407                 case action_none:
00408                 default:
00409                     break;
00410                 }
00411             }
00412 
00413 
00414     // ----- data members ---------------------------------------------------
00415 
00416     protected:
00417         /**
00418         * The referenced Object.
00419         */
00420         const T* m_cpo;
00421 
00422     // ----- friends --------------------------------------------------------
00423 
00424     /**
00425     * @internal
00426     */
00427     friend class SynchronizedMemberReadBlock;
00428 
00429     /**
00430     * @internal
00431     */
00432     friend class SynchronizedMemberWriteBlock;
00433     };
00434 
00435 
00436 // ----- non-member operators and functions ---------------------------------
00437 
00438 /**
00439 * Output a human-readable description of the given MemberView to the
00440 * specified stream.
00441 *
00442 * @param out  the stream used to output the description
00443 * @param mv   the MemberView to describe
00444 *
00445 * @return the supplied stream
00446 */
00447 template <typename Char, typename Traits, class T>
00448 COH_INLINE std::basic_ostream<Char, Traits>& operator<<(std::basic_ostream<Char, Traits>& out, const MemberView<T>& mv)
00449     {
00450     out << (typename T::View) mv;
00451     return out;
00452     }
00453 
00454 /**
00455 * Assign the specified holder to NULL.
00456 *
00457 * @param mv the MemberView to clear
00458 */
00459 template<class T> void clear_handle(MemberView<T>& mv)
00460     {
00461     mv = NULL;
00462     }
00463 
00464 /**
00465 * Perform a dynamic cast the pointer associated with the MemberView
00466 * to a the specified handle/view type.
00467 *
00468 * @param mv      the MemberView from which to perform the cast
00469 * @param fThrow  true if an exception is to be thrown on a failed cast
00470 *
00471 * @return the casted pointer, or NULL if the cast fails and fThrow is false
00472 *
00473 * @throws ClassCastException if the cast fails and fThrow is true
00474 */
00475 template<class D, class T>
00476 D cast(const MemberView<T>& mv, bool fThrow = true)
00477     {
00478     return cast<D>((typename MemberView<T>::ValueView) mv, fThrow);
00479     }
00480 
00481 /**
00482 * Perform an instanceof check on a MemberView.
00483 *
00484 * @param mv  the MemberView from which to perform the test
00485 *
00486 * @return true if the supplied handle is an instance of the specified type
00487 */
00488 template<class D, class T>
00489 bool instanceof(const MemberView<T>& mv)
00490     {
00491     return NULL != cast<D>(mv, false);
00492     }
00493 
00494 /**
00495 * Return true if the supplied holder equals NULL.
00496 *
00497 * @param mv  the MemberView to test
00498 *
00499 * @return true iff the supplied MemberView equals NULL
00500 */
00501 template<class T>
00502 bool is_null(const MemberView<T>& mv)
00503     {
00504     return mv == NULL;
00505     }
00506 
00507 COH_CLOSE_NAMESPACE2
00508 
00509 #endif // COH_MEMBER_VIEW_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.