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

F79659-03

coherence/util/DualQueue.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_DUAL_QUEUE_HPP
00008 #define COH_DUAL_QUEUE_HPP
00009 
00010 #include "coherence/lang.ns"
00011 
00012 #include "coherence/util/AbstractConcurrentQueue.hpp"
00013 #include "coherence/util/Iterator.hpp"
00014 #include "coherence/util/List.hpp"
00015 
00016 COH_OPEN_NAMESPACE2(coherence,util)
00017 
00018 
00019 /**
00020 * The DualQueue is optimized for the producer consumer use case.
00021 *
00022 * Producers work on the tail of the queue, consumers operate on the head of
00023 * the queue.  The two portions of the queue are maintained as separate lists,
00024 * and protected by separate locks.
00025 *
00026 * When a consumer looks at the head of the queue, if it is empty, the head
00027 * and tail will be swapped.
00028 *
00029 * @author nsa 2008.02.13
00030 */
00031 class COH_EXPORT DualQueue
00032     : public class_spec<DualQueue,
00033         extends<AbstractConcurrentQueue> >
00034     {
00035     friend class factory<DualQueue>;
00036 
00037     // ----- constructors ---------------------------------------------------
00038 
00039     protected:
00040         /**
00041         * Create a new DualQueue.
00042         *
00043         * @return a new DualQueue
00044         */
00045         DualQueue();
00046 
00047 
00048     // ----- DualQueue interface --------------------------------------------
00049 
00050     protected:
00051         /**
00052         * Swap the head and the tail, but only if the head is empty and the
00053         * tail is not. The calling thread must already hold a the head lock.
00054         *
00055         * @return true iff the head and tail were swapped
00056         */
00057         virtual bool swapNoWait();
00058 
00059         /**
00060         * Return the head element list.
00061         *
00062         * @return the head element list
00063         */
00064         virtual List::Handle getHeadElementList();
00065 
00066         /**
00067         * Set the head element list.
00068         *
00069         * @param hList  the new head list to set
00070         */
00071         virtual void setHeadElementList(List::Handle hList);
00072 
00073         /**
00074         * Return the element list (tail).
00075         *
00076         * @return the element list
00077         */
00078         virtual List::Handle getElementList();
00079 
00080         /**
00081         * Set the element list (tail)
00082         *
00083         * @param hList  the new list to set
00084         */
00085         virtual void setElementList(List::Handle hList);
00086 
00087         /**
00088         * Return the head lock.
00089         *
00090         * @return the head lock
00091         */
00092         virtual Object::Handle getHeadLock();
00093 
00094 
00095     // ----- Queue interface ------------------------------------------------
00096 
00097     public:
00098         /**
00099         * {@inheritDoc}
00100         */
00101         virtual bool add(Object::Holder oh);
00102 
00103         /**
00104         * {@inheritDoc}
00105         */
00106         virtual bool addHead(Object::Holder oh);
00107 
00108         /**
00109         * {@inheritDoc}
00110         */
00111         virtual bool isEmpty() const;
00112 
00113         /**
00114         * {@inheritDoc}
00115         */
00116         virtual Object::Holder peekNoWait();
00117 
00118         /**
00119         * {@inheritDoc}
00120         */
00121         virtual Object::Holder removeNoWait();
00122 
00123 
00124     // ----- data members ---------------------------------------------------
00125 
00126     protected:
00127         /**
00128         * Lock protecting operations on the head of the queue, and tail
00129         * swapping. We cannot simply lock on the head element list as it
00130         * gets swapped with the tail.
00131         *
00132         * To avoid deadlock issues the Queue lock should never be obtained
00133         * while holding the head lock.
00134         *
00135         * For example:
00136         * <pre>
00137         * COH_SYNCHRONIZED(getHeadLock())
00138         *     {
00139         *     COH_SYNCHRONIZED(this)
00140         *         {
00141         *         // this is NOT ok
00142         *         }
00143         *     }
00144         * COH_SYNCHRONIZED(this)
00145         *     {
00146         *     COH_SYNCHRONIZED(getHeadLock())
00147         *         {
00148         *         // this is ok
00149         *         }
00150         *     }
00151         * </pre>
00152         *
00153         * The latter approach was chosen as it allows users of the DualQueue
00154         * to perform external synchronization without risking deadlock.
00155         */
00156         FinalHandle<Object> f_hHeadLock;
00157 
00158         /**
00159         * The List that backs the queue. For a dual queue the ElementList is
00160         * the tail.
00161         */
00162         MemberHandle<List> m_hElementList;
00163 
00164         /**
00165         * The storage for the head of the queue.
00166         */
00167         MemberHandle<List> m_hHeadElementList;
00168     };
00169 
00170 COH_CLOSE_NAMESPACE2
00171 
00172 #endif // COH_DUAL_QUEUE_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.