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

F79659-03

coherence/lang/SynchronizedBlock.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_SYNCHRONIZED_BLOCK_HPP
00008 #define COH_SYNCHRONIZED_BLOCK_HPP
00009 
00010 #include "coherence/lang/compatibility.hpp"
00011 
00012 #include "coherence/lang/IllegalStateException.hpp"
00013 #include "coherence/lang/Object.hpp"
00014 
00015 #include <sstream>
00016 
00017 COH_OPEN_NAMESPACE2(coherence,lang)
00018 
00019 
00020 /**
00021 * The SynchronizedBlock class allows for easy creation of synchronized code
00022 * blocks based on an Object's monitor. The SynchronizedBlock object will
00023 * ensure that the monitor is entered and exited as part of starting and
00024 * ending the code block.
00025 *
00026 * Example usage:
00027 * @code
00028 * // outside of sync block
00029 *     {
00030 *     SynchronizedBlock sync(vObject);
00031 *     // monitor entered
00032 *     // critical section goes here
00033 *     // ...
00034 *     } // monitor exited
00035 * // outside of sync block
00036 * @endcode
00037 *
00038 * A more friendly form is to use the COH_SYNCHRONIZED macro. Example usage:
00039 *
00040 * @code
00041 * // outside of sync block
00042 * COH_SYNCHRONIZED(vLock) // monitor entered
00043 *   {
00044 *   // critical section goes here
00045 *   // ...
00046 *   // ...
00047 *   } // monitor exited
00048 * // outside of sync block
00049 * @endcode
00050 *
00051 * @author mf 2007.12.09
00052 */
00053 class COH_EXPORT SynchronizedBlock
00054     {
00055     // ----- constructors ---------------------------------------------------
00056 
00057     public:
00058         /**
00059         * Construct a SynchronizedBlock object.
00060         *
00061         * This will automatically enter the monitor.
00062         */
00063         SynchronizedBlock(Object::View v)
00064             : m_cpObject(get_pointer(v)), m_vObject(v)
00065             {
00066             v->_enterMonitor();
00067             }
00068 
00069         /**
00070         * Construct a SynchronizedBlock object from a raw object pointer.
00071         *
00072         * This will automatically enter the monitor.
00073         *
00074         * The caller must ensure that the referenced Object is kept alive for
00075         * the lifetime of the SynchronizedBlock.  This method is generally
00076         * used for synchronizing on "this", which implicitly ensures the
00077         * lifetime guarantee.
00078         */
00079         SynchronizedBlock(const Object* p)
00080             : m_cpObject(p), m_vObject()
00081             {
00082             if (p == NULL)
00083                 {
00084                 coh_throw_npe(typeid(Object));
00085                 }
00086             p->_enterMonitor();
00087             }
00088 
00089         /**
00090         * Copy constructor for COH_SYNCHRONIZED macro.
00091         */
00092         SynchronizedBlock(const SynchronizedBlock& that)
00093             : m_cpObject(that.m_cpObject), m_vObject(that.m_vObject)
00094             {
00095             that.m_cpObject = NULL;
00096             that.m_vObject  = NULL;
00097             }
00098 
00099         /**
00100         * Destroy a SynchronizedBlock object.
00101         *
00102         * This will automatically exit the monitor.
00103         */
00104         ~SynchronizedBlock()
00105             {
00106             const Object* cp = m_cpObject;
00107             if (NULL != cp)
00108                 {
00109                 m_cpObject = NULL;
00110                 cp->_exitMonitor();
00111                 m_vObject  = NULL;
00112                 }
00113             }
00114 
00115 
00116     // ----- operators ------------------------------------------------------
00117 
00118     public:
00119         /*
00120         * Boolean conversion for use in COH_SYNCHRONIZED macro.
00121         *
00122         * @return false if synchronization is held, true otherwise
00123         */
00124         operator bool() const
00125             {
00126             return m_cpObject == NULL;
00127             }
00128 
00129     private:
00130         /**
00131         * Blocked assignment operator.
00132         */
00133         const SynchronizedBlock& operator=(const SynchronizedBlock&);
00134 
00135         /**
00136         * Blocked dynamic allocation.
00137         */
00138         static void* operator new(size_t);
00139 
00140 
00141     // ----- data members ---------------------------------------------------
00142 
00143     protected:
00144         /**
00145         * Raw pointer to Object used to protect the synchronized block.
00146         */
00147         mutable const Object* m_cpObject;
00148 
00149         /**
00150         * Object used to protect the synchronized block.
00151         */
00152         mutable Object::View m_vObject; // on stack
00153     };
00154 
00155 COH_CLOSE_NAMESPACE2
00156 
00157 
00158 /**
00159 * Macro for making more readable synchronized code blocks See the
00160 * documentation of SynchronizedBlock for a usage example.
00161 *
00162 * @see coherence::lang::SynchronizedBlock
00163 */
00164 #define COH_SYNCHRONIZED(V) \
00165     if (coherence::lang::SynchronizedBlock COH_UNIQUE_IDENTIFIER(_coh_sync_) \
00166         = coherence::lang::SynchronizedBlock(V)) \
00167         { \
00168         COH_THROW(coherence::lang::IllegalStateException::create()); \
00169         } \
00170     else
00171 
00172 #endif // COH_SYNCHRONIZED_BLOCK_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.