lists.h
上传用户:hzhsqp
上传日期:2007-01-06
资源大小:1600k
文件大小:38k
源码类别:

IP电话/视频会议

开发平台:

Visual C++

  1. /*
  2.  * lists.h
  3.  *
  4.  * List Container Classes
  5.  *
  6.  * Portable Windows Library
  7.  *
  8.  * Copyright (c) 1993-1998 Equivalence Pty. Ltd.
  9.  *
  10.  * The contents of this file are subject to the Mozilla Public License
  11.  * Version 1.0 (the "License"); you may not use this file except in
  12.  * compliance with the License. You may obtain a copy of the License at
  13.  * http://www.mozilla.org/MPL/
  14.  *
  15.  * Software distributed under the License is distributed on an "AS IS"
  16.  * basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
  17.  * the License for the specific language governing rights and limitations
  18.  * under the License.
  19.  *
  20.  * The Original Code is Portable Windows Library.
  21.  *
  22.  * The Initial Developer of the Original Code is Equivalence Pty. Ltd.
  23.  *
  24.  * Portions are Copyright (C) 1993 Free Software Foundation, Inc.
  25.  * All Rights Reserved.
  26.  *
  27.  * Contributor(s): ______________________________________.
  28.  *
  29.  * $Log: lists.h,v $
  30.  * Revision 1.19  2000/04/14 07:19:32  craigs
  31.  * Fixed problem with assert when dequeueing from an empty queue
  32.  *
  33.  * Revision 1.18  1999/08/22 12:13:43  robertj
  34.  * Fixed warning when using inlines on older GNU compiler
  35.  *
  36.  * Revision 1.17  1999/03/09 02:59:50  robertj
  37.  * Changed comments to doc++ compatible documentation.
  38.  *
  39.  * Revision 1.16  1999/02/16 08:12:00  robertj
  40.  * MSVC 6.0 compatibility changes.
  41.  *
  42.  * Revision 1.15  1998/09/23 06:20:49  robertj
  43.  * Added open source copyright license.
  44.  *
  45.  * Revision 1.14  1997/06/08 04:49:12  robertj
  46.  * Fixed non-template class descendent order.
  47.  *
  48.  * Revision 1.13  1997/04/27 05:50:10  robertj
  49.  * DLL support.
  50.  *
  51.  * Revision 1.12  1997/02/14 13:53:59  robertj
  52.  * Major rewrite of sorted list to use sentinel record instead of NULL pointers.
  53.  *
  54.  * Revision 1.11  1996/07/15 10:32:50  robertj
  55.  * Fixed bug in sorted list (crash on remove).
  56.  *
  57.  * Revision 1.10  1996/05/26 03:25:13  robertj
  58.  * Compatibility to GNU 2.7.x
  59.  *
  60.  * Revision 1.9  1996/01/23 13:13:32  robertj
  61.  * Fixed bug in sorted list GetObjectsIndex not checking if is same object
  62.  *
  63.  * Revision 1.8  1995/08/24 12:35:00  robertj
  64.  * Added assert for list index out of bounds.
  65.  *
  66.  * Revision 1.7  1995/06/17 11:12:43  robertj
  67.  * Documentation update.
  68.  *
  69.  * Revision 1.6  1995/03/14 12:41:41  robertj
  70.  * Updated documentation to use HTML codes.
  71.  *
  72.  * Revision 1.5  1995/02/22  10:50:30  robertj
  73.  * Changes required for compiling release (optimised) version.
  74.  *
  75.  * Revision 1.4  1995/02/05  00:48:05  robertj
  76.  * Fixed template version.
  77.  *
  78.  * Revision 1.3  1995/01/15  04:49:23  robertj
  79.  * Fixed errors in template version.
  80.  *
  81.  * Revision 1.2  1994/12/21  11:53:12  robertj
  82.  * Documentation and variable normalisation.
  83.  *
  84.  * Revision 1.1  1994/12/12  09:59:35  robertj
  85.  * Initial revision
  86.  *
  87.  */
  88. #ifdef __GNUC__
  89. #pragma interface
  90. #endif
  91. ///////////////////////////////////////////////////////////////////////////////
  92. // PList container class
  93. /**This class is a collection of objects which are descendents of the
  94.    #PObject# class. It is implemeted as a doubly linked list.
  95.    The implementation of a list allows very fast inserting and deleting of
  96.    objects in the collection, but has severe penalties for random access. All
  97.    object access should be done sequentially to avoid these speed penalties.
  98.    The class remembers the last accessed element. This state information is
  99.    used to optimise access by the "virtual array" model of collections. If
  100.    access via ordinal index is made sequentially there is little overhead.
  101.    The PAbstractList class would very rarely be descended from directly by
  102.    the user. The #PDECLARE_LIST# and #PLIST# macros would normally
  103.    be used to create descendent classes. They will instantiate the template
  104.    based on #PList# or directly declare and define the class (using
  105.    inline functions) if templates are not being used.
  106.    The #PList# class or #PDECLARE_LIST# macro will define the
  107.    correctly typed operators for subscript access (#operator[]#).
  108.  */
  109. class PAbstractList : public PCollection
  110. {
  111.   PCONTAINERINFO(PAbstractList, PCollection);
  112.   public:
  113.   /**@name Construction */
  114.   //@{
  115.     /**Create a new, empty, list.
  116.        Note that by default, objects placed into the list will be deleted when
  117.        removed or when all references to the list are destroyed.
  118.      */
  119.     PINLINE PAbstractList();
  120.   //@}
  121.   // Overrides from class PObject
  122.     /**Get the relative rank of the two lists. The following algorithm is
  123.        employed for the comparison:
  124. begin{description}
  125.        item[#EqualTo#] if the two lists are identical in length
  126.        and each objects values, not pointer, are equal.
  127.        item[#LessThan#] if the instances object value at an
  128.        ordinal position is less than the corresponding objects value in the
  129.        #obj# parameters list.
  130.                           
  131.        This is also returned if all objects are equal and the instances list
  132.        length is less than the #obj# parameters list length.
  133.        item[#GreaterThan#] if the instances object value at an
  134.        ordinal position is greater than the corresponding objects value in the
  135.        #obj# parameters list.
  136.                           
  137.        This is also returned if all objects are equal and the instances list
  138.        length is greater than the #obj# parameters list length.
  139. end{description}
  140.        @return
  141.        comparison of the two objects, #EqualTo# for same,
  142.        #LessThan# for #obj# logically less than the
  143.        object and #GreaterThan# for #obj# logically
  144.        greater than the object.
  145.      */
  146.     virtual Comparison Compare(const PObject & obj) const;
  147.   /**@name Overrides from class PContainer */
  148.   //@{
  149.     /**This function is meaningless for lists. The size of the collection is
  150.        determined by the addition and removal of objects. The size cannot be
  151.        set in any other way.
  152.        @return
  153.        Always TRUE.
  154.      */
  155.     virtual BOOL SetSize(
  156.       PINDEX newSize  /// New size for the list, this is ignored.
  157.     );
  158.   //@}
  159.   /**@name Overrides from class PCollection */
  160.   //@{
  161.     /**Append a new object to the collection. This places a new link at the
  162.        "tail" of the list.
  163.     
  164.        @return
  165.        index of the newly added object.
  166.      */
  167.     virtual PINDEX Append(
  168.       PObject * obj   /// New object to place into the collection.
  169.     );
  170.     /**Insert a new object immediately before the specified object. If the
  171.        object to insert before is not in the collection then the equivalent of
  172.        the #Append()# function is performed.
  173.        
  174.        Note that the object values are compared for the search of the
  175.        #before# parameter, not the pointers. So the objects in the
  176.        collection must correctly implement the #PObject::Compare()#
  177.        function.
  178.        @return
  179.        index of the newly inserted object.
  180.      */
  181.     virtual PINDEX Insert(
  182.       const PObject & before,   /// Object value to insert before.
  183.       PObject * obj             /// New object to place into the collection.
  184.     );
  185.     /**Insert a new object at the specified ordinal index. If the index is
  186.        greater than the number of objects in the collection then the
  187.        equivalent of the #Append()# function is performed.
  188.        @return
  189.        index of the newly inserted object.
  190.      */
  191.     virtual PINDEX InsertAt(
  192.       PINDEX index,   /// Index position in collection to place the object.
  193.       PObject * obj   /// New object to place into the collection.
  194.     );
  195.     /**Remove the object from the collection. If the AllowDeleteObjects option
  196.        is set then the object is also deleted.
  197.        @return
  198.        TRUE if the object was in the collection.
  199.      */
  200.     virtual BOOL Remove(
  201.       const PObject * obj   /// Existing object to remove from the collection.
  202.     );
  203.     /**Remove the object at the specified ordinal index from the collection.
  204.        If the AllowDeleteObjects option is set then the object is also deleted.
  205.        Note if the index is beyond the size of the collection then the
  206.        function will assert.
  207.        @return
  208.        pointer to the object being removed, or NULL if it was deleted.
  209.      */
  210.     virtual PObject * RemoveAt(
  211.       PINDEX index   /// Index position in collection to place the object.
  212.     );
  213.     /**Set the object at the specified ordinal position to the new value. This
  214.        will overwrite the existing entry. If the AllowDeleteObjects option is
  215.        set then the old object is also deleted.
  216.        The object accessed in this way is remembered by the class and further
  217.        access will be fast. Access to elements one either side of that saved
  218.        element, and the head and tail of the list, will always be fast.
  219.        Note if the index is beyond the size of the collection then the
  220.        function will assert.
  221.        @return
  222.        TRUE if the object was successfully added.
  223.      */
  224.     virtual BOOL SetAt(
  225.       PINDEX index,   /// Index position in collection to set.
  226.       PObject * val   /// New value to place into the collection.
  227.     );
  228.     /**Get the object at the specified ordinal position. If the index was
  229.        greater than the size of the collection then NULL is returned.
  230.        The object accessed in this way is remembered by the class and further
  231.        access will be fast. Access to elements one either side of that saved
  232.        element, and the head and tail of the list, will always be fast.
  233.        @return
  234.        pointer to object at the specified index.
  235.      */
  236.     virtual PObject * GetAt(
  237.       PINDEX index  // Index position in the collection of the object.
  238.     ) const;
  239.     /**Search the collection for the specific instance of the object. The
  240.        object pointers are compared, not the values. A simple linear search
  241.        from "head" of the list is performed.
  242.        @return
  243.        ordinal index position of the object, or P_MAX_INDEX.
  244.      */
  245.     virtual PINDEX GetObjectsIndex(
  246.       const PObject * obj  /// Object to find.
  247.     ) const;
  248.     /**Search the collection for the specified value of the object. The object
  249.        values are compared, not the pointers.  So the objects in the
  250.        collection must correctly implement the #PObject::Compare()#
  251.        function. A simple linear search from "head" of the list is performed.
  252.        @return
  253.        ordinal index position of the object, or P_MAX_INDEX.
  254.      */
  255.     virtual PINDEX GetValuesIndex(
  256.       const PObject & obj  /// Object to find value of.
  257.     ) const;
  258.   //@}
  259.   protected:
  260.     /**Get the object at the specified ordinal position. If the index was
  261.        greater than the size of the collection then this asserts.
  262.        The object accessed in this way is remembered by the class and further
  263.        access will be fast. Access to elements one either side of that saved
  264.        element, and the head and tail of the list, will always be fast.
  265.        @return
  266.        reference to object at the specified index.
  267.      */
  268.     PINLINE PObject & GetReferenceAt(
  269.       PINDEX index  /// Ordinal index of the list element to set as current.
  270.     ) const;
  271.     /**Move the internal "cursor" to the index position specified. This
  272.        function will optimise the sequential move taking into account the
  273.        previous current position and the position at the head and tail of the
  274.        list. Whichever of these three points is closes is used as the starting
  275.        point for a sequential move to the required index.
  276.        @return
  277.        TRUE if the index could be set as the current element.
  278.      */
  279.     BOOL SetCurrent(
  280.       PINDEX index  /// Ordinal index of the list element to set as current.
  281.     ) const;
  282.     class Element {
  283.       public:
  284.         Element(PObject * theData);
  285.         Element * prev;
  286.         Element * next;
  287.         PObject * data;
  288.     };
  289.     class Info {
  290.       public:
  291.         Info() { head = tail = lastElement = NULL; }
  292.         Element * head;
  293.         Element * tail;
  294.         Element * lastElement;
  295.         PINDEX    lastIndex;
  296.     } * info;
  297. };
  298. #ifdef PHAS_TEMPLATES
  299. /**This template class maps the PAbstractList to a specific object type. The
  300.    functions in this class primarily do all the appropriate casting of types.
  301.    Note that if templates are not used the #PDECLARE_LIST# macro will
  302.    simulate the template instantiation.
  303.  */
  304. template <class T> class PList : public PAbstractList
  305. {
  306.   PCLASSINFO(PList, PAbstractList);
  307.   public:
  308.   /**@name Construction */
  309.   //@{
  310.     /**Create a new, empty, list.
  311.        Note that by default, objects placed into the list will be deleted when
  312.        removed or when all references to the list are destroyed.
  313.      */
  314.     PList()
  315.       : PAbstractList() { }
  316.   //@}
  317.   /**@name Overrides from class PObject */
  318.   //@{
  319.     /**Make a complete duplicate of the list. Note that all objects in the
  320.        array are also cloned, so this will make a complete copy of the list.
  321.      */
  322.     virtual PObject * Clone() const
  323.       { return PNEW PList(0, this); }
  324.   //@}
  325.   /**@name New functions for class */
  326.   //@{
  327.     /**Retrieve a reference  to the object in the list. If there was not an
  328.        object at that ordinal position or the index was beyond the size of the
  329.        array then the function asserts.
  330.        The object accessed in this way is remembered by the class and further
  331.        access will be fast. Access to elements one either side of that saved
  332.        element, and the head and tail of the list, will always be fast.
  333.        @return
  334.        reference to the object at #index# position.
  335.      */
  336.     T & operator[](PINDEX index) const
  337.       { return (T &)GetReferenceAt(index); }
  338.   //@}
  339.   protected:
  340.     PList(int dummy, const PList * c)
  341.       : PAbstractList(dummy, c) { }
  342. };
  343. /**Declare a list class.
  344.    This macro is used to declare a descendent of PAbstractList class,
  345.    customised for a particular object type {bf T}. This macro closes the
  346.    class declaration off so no additional members can be added.
  347.    If the compilation is using templates then this macro produces a typedef
  348.    of the #PList# template class.
  349.    See the #PList# class and #PDECLARE_LIST# macro for more
  350.    information.
  351.  */
  352. #define PLIST(cls, T) typedef PList<T> cls
  353. /**Begin declaration of list class.
  354.    This macro is used to declare a descendent of PAbstractList class,
  355.    customised for a particular object type {bf T}.
  356.    If the compilation is using templates then this macro produces a descendent
  357.    of the #PList# template class. If templates are not being used then the
  358.    macro defines a set of inline functions to do all casting of types. The
  359.    resultant classes have an identical set of functions in either case.
  360.    See the #PList# and #PAbstractList# classes for more information.
  361.  */
  362. #define PDECLARE_LIST(cls, T) 
  363.   PLIST(cls##_PTemplate, T); 
  364.   PDECLARE_CLASS(cls, cls##_PTemplate) 
  365.   protected: 
  366.     cls(int dummy, const cls * c) 
  367.       : cls##_PTemplate(dummy, c) { } 
  368.   public: 
  369.     cls() 
  370.       : cls##_PTemplate() { } 
  371.     virtual PObject * Clone() const 
  372.       { return PNEW cls(0, this); } 
  373. /**This template class maps the PAbstractList to a specific object type, and
  374.    adds functionality that allows the list to be used as a first in first out
  375.    queue. The functions in this class primarily do all the appropriate casting
  376.    of types.
  377.    By default, objects placed into the set will {bf not} be deleted when
  378.    removed or when all references to the set are destroyed. This is different
  379.    from the default on most collection classes.
  380.    Note that if templates are not used the #PDECLARE_QUEUE# macro will
  381.    simulate the template instantiation.
  382.  */
  383. template <class T> class PQueue : public PAbstractList
  384. {
  385.   PCLASSINFO(PQueue, PAbstractList);
  386.   public:
  387.   /**@name Construction */
  388.   //@{
  389.     /**Create a new, empty, queue.
  390.        Note that by default, objects placed into the queue will {bf not} be
  391.        deleted when removed or when all references to the queue are destroyed.
  392.        This is different from the default on most collection classes.
  393.      */
  394.     PQueue()
  395.       : PAbstractList() { DisallowDeleteObjects(); }
  396.   //@}
  397.   /**@name Overrides from class PObject */
  398.   //@{
  399.     /**Make a complete duplicate of the list. Note that all objects in the
  400.        array are also cloned, so this will make a complete copy of the list.
  401.      */
  402.     virtual PObject * Clone() const
  403.       { return PNEW PQueue(0, this); }
  404.   //@}
  405.   /**@name New functions for class */
  406.   //@{
  407.     /**Add a new object to the queue. This places a new link at the "tail" of
  408.        the list, which is the "in" side of the queue.
  409.      */
  410.     virtual void Enqueue(
  411.       T * obj   /// Object to add to the queue.
  412.     ) { PAbstractList::Append(obj); }
  413.     /**Remove an object that was added to the queue.
  414.        @return
  415.        first object added to the queue or NULL if queue empty.
  416.      */
  417.     virtual T * Dequeue()
  418.       { if (GetSize() == 0) return NULL; else return (T *)PAbstractList::RemoveAt(0);}
  419.   //@}
  420.   protected:
  421.     PQueue(int dummy, const PQueue * c)
  422.       : PAbstractList(dummy, c)
  423.       { reference->deleteObjects = c->reference->deleteObjects; }
  424. };
  425. /**Declare a queue class.
  426.    This macro is used to declare a descendent of PAbstractList class,
  427.    customised for a particular object type {bf T}, and adds functionality
  428.    that allows the list to be used as a first in first out queue. This macro
  429.    closes the class declaration off so no additional members can be added.
  430.    If the compilation is using templates then this macro produces a typedef
  431.    of the #PQueue# template class.
  432.    See the #PList# class and #PDECLARE_QUEUE# macro for more
  433.    information.
  434.  */
  435. #define PQUEUE(cls, T) typedef PQueue<T> cls
  436. /**Begin declataion of a queue class.
  437.    This macro is used to declare a descendent of PAbstractList class,
  438.    customised for a particular object type {bf T}, and adds functionality
  439.    that allows the list to be used as a first in first out queue.
  440.    If the compilation is using templates then this macro produces a descendent
  441.    of the #PQueue# template class. If templates are not being used then
  442.    the macro defines a set of inline functions to do all casting of types. The
  443.    resultant classes have an identical set of functions in either case.
  444.    See the #PQueue# and #PAbstractList# classes for more information.
  445.  */
  446. #define PDECLARE_QUEUE(cls, T) 
  447.   PQUEUE(cls##_PTemplate, T); 
  448.   PDECLARE_CLASS(cls, cls##_PTemplate) 
  449.   protected: 
  450.     cls(int dummy, const cls * c) 
  451.       : cls##_PTemplate(dummy, c) { } 
  452.   public: 
  453.     cls() 
  454.       : cls##_PTemplate() { } 
  455.     virtual PObject * Clone() const 
  456.       { return PNEW cls(0, this); } 
  457. /**This template class maps the PAbstractList to a specific object type, and
  458.    adds functionality that allows the list to be used as a last in first out
  459.    stack. The functions in this class primarily do all the appropriate casting
  460.    of types.
  461.    By default, objects placed into the set will {bf not} be deleted when
  462.    removed or when all references to the set are destroyed. This is different
  463.    from the default on most collection classes.
  464.    Note that if templates are not used the #PDECLARE_STACK# macro will
  465.    simulate the template instantiation.
  466.  */
  467. template <class T> class PStack : public PAbstractList
  468. {
  469.   PCLASSINFO(PStack, PAbstractList);
  470.   public:
  471.   /**@name Construction */
  472.   //@{
  473.     /**Create a new, empty, stack.
  474.        Note that by default, objects placed into the stack will {bf not} be
  475.        deleted when removed or when all references to the stack are destroyed.
  476.        This is different from the default on most collection classes.
  477.      */
  478.     PStack()
  479.       : PAbstractList() { DisallowDeleteObjects(); }
  480.   //@}
  481.   /**@name Overrides from class PObject */
  482.   //@{
  483.     /**Make a complete duplicate of the stack. Note that all objects in the
  484.        array are also cloned, so this will make a complete copy of the stack.
  485.      */
  486.     virtual PObject * Clone() const
  487.       { return PNEW PStack(0, this); }
  488.   //@}
  489.   /**@name New functions for class */
  490.   //@{
  491.     /**Add an object to the stack. This object will be on "top" of the stack
  492.        and will be the object returned by the #Pop()#
  493.        function.
  494.      */
  495.     virtual void Push(
  496.       T * obj    /// Object to add to the stack.
  497.     ) { PAbstractList::InsertAt(0, obj); }
  498.     /**Remove the last object pushed onto the stack.
  499.        @return
  500.        object on top of the stack.
  501.      */
  502.     virtual T * Pop()
  503.       { return (T *)PAbstractList::RemoveAt(0); }
  504.     /**Get the element that is currently on top of the stack without removing
  505.        it.
  506.        @return
  507.        reference to object on top of the stack.
  508.      */
  509.     virtual T & Top()
  510.       { PAssert(GetSize() > 0, PStackEmpty); return *(T *)GetAt(0); }
  511.   //@}
  512.   protected:
  513.     PStack(int dummy, const PStack * c)
  514.       : PAbstractList(dummy, c)
  515.       { reference->deleteObjects = c->reference->deleteObjects; }
  516. };
  517. /**Declare a stack class.
  518.    This macro is used to declare a descendent of PAbstractList class,
  519.    customised for a particular object type {bf T}, and adds functionality
  520.    that allows the list to be used as a last in first out stack. This macro
  521.    closes the class declaration off so no additional members can be added.
  522.    If the compilation is using templates then this macro produces a typedef
  523.    of the #PStack# template class.
  524.    See the #PStack# class and #PDECLARE_STACK# macro for more
  525.    information.
  526.  */
  527. #define PSTACK(cls, T) typedef PStack<T> cls
  528. /**Begin declaration of a stack class.
  529.    This macro is used to declare a descendent of PAbstractList class,
  530.    customised for a particular object type {bf T}, and adds functionality
  531.    that allows the list to be used as a last in first out stack.
  532.    If the compilation is using templates then this macro produces a descendent
  533.    of the #PStack# template class. If templates are not being used then
  534.    the macro defines a set of inline functions to do all casting of types. The
  535.    resultant classes have an identical set of functions in either case.
  536.    See the #PStack# and #PAbstractList# classes for more information.
  537.  */
  538. #define PDECLARE_STACK(cls, T) 
  539.   PSTACK(cls##_PTemplate, T); 
  540.   PDECLARE_CLASS(cls, cls##_PTemplate) 
  541.   protected: 
  542.     cls(int dummy, const cls * c) 
  543.       : cls##_PTemplate(dummy, c) { } 
  544.   public: 
  545.     cls() 
  546.       : cls##_PTemplate() { } 
  547.     virtual PObject * Clone() const 
  548.       { return PNEW cls(0, this); } 
  549. #else // PHAS_TEMPLATES
  550. #define PLIST(cls, T) 
  551.   class cls : public PAbstractList { 
  552.   PCLASSINFO(cls, PAbstractList); 
  553.   protected: 
  554.     inline cls(int dummy, const cls * c) 
  555.       : PAbstractList(dummy, c) { } 
  556.   public: 
  557.     inline cls() 
  558.       : PAbstractList() { } 
  559.     inline virtual PObject * Clone() const 
  560.       { return PNEW cls(0, this); } 
  561.     inline T & operator[](PINDEX index) const 
  562.       { return (T &)GetReferenceAt(index); } 
  563.   }
  564. #define PDECLARE_LIST(cls, T) 
  565.   PLIST(cls##_PTemplate, T); 
  566.   PDECLARE_CLASS(cls, cls##_PTemplate) 
  567.   protected: 
  568.     cls(int dummy, const cls * c) 
  569.       : cls##_PTemplate(dummy, c) { } 
  570.   public: 
  571.     cls() 
  572.       : cls##_PTemplate() { } 
  573.     virtual PObject * Clone() const 
  574.       { return PNEW cls(0, this); } 
  575. #define PQUEUE(cls, T) 
  576.   class cls : public PAbstractList { 
  577.   PCLASSINFO(cls, PAbstractList); 
  578.   protected: 
  579.     inline cls(int dummy, const cls * c) 
  580.       : PAbstractList(dummy, c) 
  581.       { reference->deleteObjects = c->reference->deleteObjects; } 
  582.   public: 
  583.     inline cls() 
  584.       : PAbstractList() { DisallowDeleteObjects(); } 
  585.     inline virtual PObject * Clone() const 
  586.       { return PNEW cls(0, this); } 
  587.     virtual inline void Enqueue(T * t) 
  588.       { PAbstractList::Append(t); } 
  589.     virtual inline T * Dequeue() 
  590.       { if (GetSize() == 0) return NULL; else return (T *)PAbstractList::RemoveAt(0);} 
  591.   }
  592. #define PDECLARE_QUEUE(cls, T) 
  593.   PQUEUE(cls##_PTemplate, T); 
  594.   PDECLARE_CLASS(cls, cls##_PTemplate) 
  595.   protected: 
  596.     cls(int dummy, const cls * c) 
  597.       : cls##_PTemplate(dummy, c) { } 
  598.   public: 
  599.     cls() 
  600.       : cls##_PTemplate() { } 
  601.     virtual PObject * Clone() const 
  602.       { return PNEW cls(0, this); } 
  603. #define PSTACK(cls, T) 
  604.   class cls : public PAbstractList { 
  605.   PCLASSINFO(cls, PAbstractList); 
  606.   protected: 
  607.     inline cls(int dummy, const cls * c) 
  608.       : PAbstractList(dummy, c) 
  609.       { reference->deleteObjects = c->reference->deleteObjects; } 
  610.   public: 
  611.     inline cls() 
  612.       : PAbstractList() { DisallowDeleteObjects(); } 
  613.     inline virtual PObject * Clone() const 
  614.       { return PNEW cls(0, this); } 
  615.     virtual inline void Push(T * t) 
  616.       { PAbstractList::InsertAt(0, t); } 
  617.     virtual inline T * Pop() 
  618.       { PAssert(GetSize() > 0, PStackEmpty); return (T *)PAbstractList::RemoveAt(0); } 
  619.     virtual inline T & Top() 
  620.       { PAssert(GetSize() > 0, PStackEmpty); return *(T *)GetAt(0); } 
  621.   }
  622. #define PDECLARE_STACK(cls, T) 
  623.   PSTACK(cls##_PTemplate, T); 
  624.   PDECLARE_CLASS(cls, cls##_PTemplate) 
  625.   protected: 
  626.     cls(int dummy, const cls * c) 
  627.       : cls##_PTemplate(dummy, c) { } 
  628.   public: 
  629.     cls() 
  630.       : cls##_PTemplate() { } 
  631.     virtual PObject * Clone() const 
  632.       { return PNEW cls(0, this); } 
  633. #endif // PHAS_TEMPLATES
  634. ///////////////////////////////////////////////////////////////////////////////
  635. // Sorted List of PObjects
  636. /**This class is a collection of objects which are descendents of the
  637.    #PObject# class. It is implemeted as a Red-Black binary tree to
  638.    maintain the objects in rank order. Note that this requires that the
  639.    #PObject::Compare()# function be fully implemented oin objects
  640.    contained in the collection.
  641.    The implementation of a sorted list allows fast inserting and deleting as
  642.    well as random access of objects in the collection. As the objects are being
  643.    kept sorted, "fast" is a relative term. All operations take o(lg n) unless
  644.    a particular object is repeatedly accessed.
  645.    The class remembers the last accessed element. This state information is
  646.    used to optimise access by the "virtual array" model of collections. If
  647.    repeated access via ordinal index is made there is little overhead. All
  648.    other access incurs a minimum overhead, but not insignificant.
  649.    The PAbstractSortedList class would very rarely be descended from directly
  650.    by the user. The #PDECLARE_LIST# and #PLIST# macros would normally
  651.    be used to create descendent classes. They will instantiate the template
  652.    based on #PSortedList# or directly declare and define the class (using
  653.    inline functions) if templates are not being used.
  654.    The #PSortedList# class or #PDECLARE_SORTED_LIST# macro will
  655.    define the correctly typed operators for subscript access
  656.    (#operator[]#).
  657.  */
  658. class PAbstractSortedList : public PCollection
  659. {
  660.   PCONTAINERINFO(PAbstractSortedList, PCollection);
  661.   public:
  662.   /**@name Construction */
  663.   //@{
  664.     /**Create a new, empty, sorted list.
  665.        Note that by default, objects placed into the list will be deleted when
  666.        removed or when all references to the list are destroyed.
  667.      */
  668.     PAbstractSortedList();
  669.   //@}
  670.   /**@name Overrides from class PObject */
  671.   //@{
  672.     /**Get the relative rank of the two lists. The following algorithm is
  673.        employed for the comparison:
  674. begin{descriptions}
  675.        item[#EqualTo#] if the two lists are identical in length
  676.        and each objects values, not pointer, are equal.
  677.        item[#LessThan#] if the instances object value at an
  678.        ordinal position is less than the corresponding objects value in the
  679.        #obj# parameters list.
  680.                           
  681.        This is also returned if all objects are equal and the instances list
  682.        length is less than the #obj# parameters list length.
  683.        item[#GreaterThan#] if the instances object value at an
  684.        ordinal position is greater than the corresponding objects value in the
  685.        #obj# parameters list.
  686.                           
  687.        This is also returned if all objects are equal and the instances list
  688.        length is greater than the #obj# parameters list length.
  689. end{descriptions}
  690.        @return
  691.        comparison of the two objects, #EqualTo# for same,
  692.        #LessThan# for #obj# logically less than the
  693.        object and #GreaterThan# for #obj# logically
  694.        greater than the object.
  695.      */
  696.     virtual Comparison Compare(const PObject & obj) const;
  697.   //@}
  698.   /**@name Overrides from class PContainer */
  699.   //@{
  700.     /**This function is meaningless for lists. The size of the collection is
  701.        determined by the addition and removal of objects. The size cannot be
  702.        set in any other way.
  703.        @return
  704.        Always TRUE.
  705.      */
  706.     virtual BOOL SetSize(
  707.       PINDEX newSize  // New size for the sorted list, this is ignored.
  708.     );
  709.   //@}
  710.   /**@name Overrides from class PCollection */
  711.   //@{
  712.     /**Add a new object to the collection. The object is always placed in the
  713.        correct ordinal position in the list. It is not placed at the "end".
  714.        @return
  715.        index of the newly added object.
  716.      */
  717.     virtual PINDEX Append(
  718.       PObject * obj   // New object to place into the collection.
  719.     );
  720.     /**Add a new object to the collection.
  721.     
  722.        The object is always placed in the correct ordinal position in the list.
  723.        It is not placed at the specified position. The #before#
  724.        parameter is ignored.
  725.        @return
  726.        index of the newly inserted object.
  727.      */
  728.     virtual PINDEX Insert(
  729.       const PObject & before,   // Object value to insert before.
  730.       PObject * obj             // New object to place into the collection.
  731.     );
  732.     /**Add a new object to the collection.
  733.     
  734.        The object is always placed in the correct ordinal position in the list.
  735.        It is not placed at the specified position. The #index#
  736.        parameter is ignored.
  737.        @return
  738.        index of the newly inserted object.
  739.      */
  740.     virtual PINDEX InsertAt(
  741.       PINDEX index,   // Index position in collection to place the object.
  742.       PObject * obj   // New object to place into the collection.
  743.     );
  744.     /**Remove the object from the collection. If the AllowDeleteObjects option
  745.        is set then the object is also deleted.
  746.        Note that the comparison for searching for the object in collection is
  747.        made by pointer, not by value. Thus the parameter must point to the
  748.        same instance of the object that is in the collection.
  749.        @return
  750.        TRUE if the object was in the collection.
  751.      */
  752.     virtual BOOL Remove(
  753.       const PObject * obj   // Existing object to remove from the collection.
  754.     );
  755.     /**Remove the object at the specified ordinal index from the collection.
  756.        If the AllowDeleteObjects option is set then the object is also deleted.
  757.        Note if the index is beyond the size of the collection then the
  758.        function will assert.
  759.        @return
  760.        pointer to the object being removed, or NULL if it was deleted.
  761.      */
  762.     virtual PObject * RemoveAt(
  763.       PINDEX index   // Index position in collection to place the object.
  764.     );
  765.     /**Remove all of the elements in the collection. This operates by
  766.        continually calling #RemoveAt()# until there are no objects left.
  767.        The objects are removed from the last, at index
  768.        #(GetSize()-1)# toward the first at index zero.
  769.      */
  770.     virtual void RemoveAll();
  771.     /**Set the object at the specified ordinal position to the new value. This
  772.        will overwrite the existing entry. If the AllowDeleteObjects option is
  773.        set then the old object is also deleted.
  774.        Note, the object placed at #index# will not stay at that
  775.        ordinal position. It is actually placed at the correct position for its
  776.        rank.
  777.        @return
  778.        TRUE if the object was successfully added.
  779.      */
  780.     virtual BOOL SetAt(
  781.       PINDEX index,   // Index position in collection to set.
  782.       PObject * val   // New value to place into the collection.
  783.     );
  784.     /**Get the object at the specified ordinal position. If the index was
  785.        greater than the size of the collection then NULL is returned.
  786.        @return
  787.        pointer to object at the specified index.
  788.      */
  789.     virtual PObject * GetAt(
  790.       PINDEX index  // Index position in the collection of the object.
  791.     ) const;
  792.     /**Search the collection for the specific instance of the object. The
  793.        object pointers are compared, not the values. A binary search is
  794.        employed to locate the entry.
  795.        
  796.        Note that that will require value comparisons to be made to find the
  797.        equivalent entry and then a final check is made with the pointers to
  798.        see if they are the same instance.
  799.        @return
  800.        ordinal index position of the object, or P_MAX_INDEX.
  801.      */
  802.     virtual PINDEX GetObjectsIndex(
  803.       const PObject * obj
  804.     ) const;
  805.     /**Search the collection for the specified value of the object. The object
  806.        values are compared, not the pointers.  So the objects in the
  807.        collection must correctly implement the #PObject::Compare()#
  808.        function. A binary search is employed to locate the entry.
  809.        @return
  810.        ordinal index position of the object, or P_MAX_INDEX.
  811.      */
  812.     virtual PINDEX GetValuesIndex(
  813.       const PObject & obj
  814.     ) const;
  815.   //@}
  816.     class Element {
  817.       public:
  818.         Element(PObject * theData);
  819.       private:
  820.         void DeleteSubTrees(BOOL deleteObject);
  821.         Element * Successor() const;
  822.         Element * Predecessor() const;
  823.         Element * OrderSelect(PINDEX index);
  824.         PINDEX ValueSelect(const PObject & obj, Element * & lastElement);
  825.         Element * parent;
  826.         Element * left;
  827.         Element * right;
  828.         PObject * data;
  829.         PINDEX subTreeSize;
  830.         enum { Red, Black } colour;
  831.       friend class PAbstractSortedList;
  832.     };
  833.     friend class Element;
  834.   protected:
  835.     class Info {
  836.       public:
  837.         Info();
  838.         Element * root;
  839.         Element * lastElement;
  840.         PINDEX    lastIndex;
  841.     } * info;
  842.     friend class Info;
  843.     // New functions for class
  844.     void RemoveElement(Element * node);
  845.     void LeftRotate(Element * node);
  846.     void RightRotate(Element * node);
  847. };
  848. #ifdef PHAS_TEMPLATES
  849. /**This template class maps the PAbstractSortedList to a specific object type.
  850.    The functions in this class primarily do all the appropriate casting of
  851.    types.
  852.    Note that if templates are not used the #PDECLARE_SORTED_LIST# macro
  853.    will simulate the template instantiation.
  854.  */
  855. template <class T> class PSortedList : public PAbstractSortedList
  856. {
  857.   PCLASSINFO(PSortedList, PAbstractSortedList);
  858.   public:
  859.   /**@name Construction */
  860.   //@{
  861.     /**Create a new, empty, sorted list.
  862.        Note that by default, objects placed into the list will be deleted when
  863.        removed or when all references to the list are destroyed.
  864.      */
  865.     PSortedList()
  866.       : PAbstractSortedList() { }
  867.   //@}
  868.   /**@name Overrides from class PObject */
  869.   //@{
  870.     /**Make a complete duplicate of the list. Note that all objects in the
  871.        array are also cloned, so this will make a complete copy of the list.
  872.      */
  873.     virtual PObject * Clone() const
  874.       { return PNEW PSortedList(0, this); }
  875.   //@}
  876.   /**@name New functions for class */
  877.   //@{
  878.     /**Retrieve a reference  to the object in the list. If there was not an
  879.        object at that ordinal position or the index was beyond the size of the
  880.        array then the function asserts.
  881.        The object accessed in this way is remembered by the class and further
  882.        access will be fast.
  883.        @return
  884.        reference to the object at #index# position.
  885.      */
  886.     T & operator[](PINDEX index) const
  887.       { return *(T *)GetAt(index); }
  888.   //@}
  889.   protected:
  890.     PSortedList(int dummy, const PSortedList * c)
  891.       : PAbstractSortedList(dummy, c) { }
  892. };
  893. /**Declare a sorted list class.
  894.    This macro is used to declare a descendent of PAbstractSortedList class,
  895.    customised for a particular object type {bf T}. This macro closes the
  896.    class declaration off so no additional members can be added.
  897.    If the compilation is using templates then this macro produces a typedef
  898.    of the #PSortedList# template class.
  899.    See the #PSortedList# class and #PDECLARE_SORTED_LIST# macro for
  900.    more information.
  901.  */
  902. #define PSORTED_LIST(cls, T) typedef PSortedList<T> cls
  903. /**Begin declaration of a sorted list class.
  904.    This macro is used to declare a descendent of PAbstractSortedList class,
  905.    customised for a particular object type {bf T}.
  906.    If the compilation is using templates then this macro produces a descendent
  907.    of the #PSortedList# template class. If templates are not being used
  908.    then the macro defines a set of inline functions to do all casting of types.
  909.    The resultant classes have an identical set of functions in either case.
  910.    See the #PSortedList# and #PAbstractSortedList# classes for more
  911.    information.
  912.  */
  913. #define PDECLARE_SORTED_LIST(cls, T) 
  914.   PSORTED_LIST(cls##_PTemplate, T); 
  915.   PDECLARE_CLASS(cls, cls##_PTemplate) 
  916.   protected: 
  917.     cls(int dummy, const cls * c) 
  918.       : cls##_PTemplate(dummy, c) { } 
  919.   public: 
  920.     cls() 
  921.       : cls##_PTemplate() { } 
  922.     virtual PObject * Clone() const 
  923.       { return PNEW cls(0, this); } 
  924. #else // PHAS_TEMPLATES
  925. #define PSORTED_LIST(cls, T) 
  926.   class cls : public PAbstractSortedList { 
  927.   PCLASSINFO(cls, PAbstractSortedList); 
  928.   protected: 
  929.     inline cls(int dummy, const cls * c) 
  930.       : PAbstractSortedList(dummy, c) { } 
  931.   public: 
  932.     inline cls() 
  933.       : PAbstractSortedList() { } 
  934.     inline virtual PObject * Clone() const 
  935.       { return PNEW cls(0, this); } 
  936.     inline T & operator[](PINDEX index) const 
  937.       { return *(T *)GetAt(index); } 
  938.   }
  939. #define PDECLARE_SORTED_LIST(cls, T) 
  940.   PSORTED_LIST(cls##_PTemplate, T); 
  941.   PDECLARE_CLASS(cls, cls##_PTemplate) 
  942.   protected: 
  943.     cls(int dummy, const cls * c) 
  944.       : cls##_PTemplate(dummy, c) { } 
  945.   public: 
  946.     cls() 
  947.       : cls##_PTemplate() { } 
  948.     virtual PObject * Clone() const 
  949.       { return PNEW cls(0, this); } 
  950. #endif  // PHAS_TEMPLATES
  951. // End Of File ///////////////////////////////////////////////////////////////