SoPlex Doxygen Documentation
idlist.h
Go to the documentation of this file.
1 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2 /* */
3 /* This file is part of the class library */
4 /* SoPlex --- the Sequential object-oriented simPlex. */
5 /* */
6 /* Copyright (C) 1996-2012 Konrad-Zuse-Zentrum */
7 /* fuer Informationstechnik Berlin */
8 /* */
9 /* SoPlex is distributed under the terms of the ZIB Academic Licence. */
10 /* */
11 /* You should have received a copy of the ZIB Academic License */
12 /* along with SoPlex; see the file COPYING. If not email to soplex@zib.de. */
13 /* */
14 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
15 
16 /**@file idlist.h
17  * @brief Generic Real linked list.
18  */
19 #ifndef _IDLIST_H_
20 #define _IDLIST_H_
21 
22 #include <assert.h>
23 
24 #include "spxdefines.h"
25 #include "islist.h"
26 
27 namespace soplex
28 {
29 //---------------------------------------------------------------------
30 // class IdElement<T>
31 //---------------------------------------------------------------------
32 
33 /**@brief Elements for \ref soplex::IdList "IdList"s.
34  @ingroup Elementary
35 
36  #IdElement%s are derived from the template parameter class T and can hence
37  be used as such. The additional methods next() and prev() provide access
38  to the links for the list. They may freely be used by the programmer as long
39  as an IdElement is not member of a IdList. In this case, the IdList
40  controls members next() and prev(). However, IdList should provide
41  enough functionality for the user not to require any modification to these
42  members.
43  */
44 /* The use of this->the_last and this->the_first instead of just the_last
45  * and the_first is bcause the HP aCC Compiler claims that according to the
46  * Standard these otherwise could not be seen. And since I was not able to
47  * even identify a hint on this in the Draft Standard we just do it, so
48  * the HP compiler is happy since it will not hurt the others.
49  */
50 template < class T >
51 class IdElement : public T
52 {
53  //--------------------------
54  /**@name Data */
55  //@{
56  IdElement<T>* theprev; ///< pointer to previous element in the IdList
57  IdElement<T>* thenext; ///< pointer to next element in the IdList
58  //@}
59 
60 public:
61 
62  //---------------------------------------
63  /**@name Successors and predecessors */
64  //@{
65  /// returns the next element in the IdList (writeable).
67  {
68  return thenext;
69  }
70  /// returns the next element in the IdList.
71  IdElement<T>* const& next() const
72  {
73  return thenext;
74  }
75 
76  /// returns the previous element in the IdList (writeable).
78  {
79  return theprev;
80  }
81  /// returns the previous element in the IdList.
82  IdElement<T>* const& prev() const
83  {
84  return theprev;
85  }
86  //@}
87 
88  //---------------------------------------
89  /**@name Construction / destruction */
90  //@{
91  /// default constructor.
93  : theprev(0)
94  , thenext(0)
95  {}
96 
97  /// copy constructor.
98  /** Only the element itself is copied, while the links to the previous and
99  the next list element are set to zero pointers.
100  */
101  IdElement(const T& old)
102  : T(old)
103  , theprev(0)
104  , thenext(0)
105  {}
106 };
107 
108 //---------------------------------------------------------------------
109 // class IdList<T>
110 //---------------------------------------------------------------------
111 
112 
113 /**@brief Generic Real linked list.
114  @ingroup Elementary
115 
116  Class IdList implements an intrusive Real linked list as a template
117  class. As such, the list elements must provide the links themselfs. For
118  conveniance, we also provide class IdElement that adds both links to an
119  arbitrary class as template parameter.
120  */
121 template < class T >
122 class IdList : public IsList<T>
123 {
124 public:
125 
126  //---------------------------------------
127  /**@name Access */
128  //@{
129  /// returns first element in list.
130  T* first() const
131  {
132  return static_cast<T*>(this->the_first);
133  }
134 
135  /// returns last element in list.
136  T* last() const
137  {
138  return static_cast<T*>(this->the_last);
139  }
140 
141  /// returns successor of \p elem or 0, if \p elem is the last element.
142  T* next(const T *elem) const
143  {
144  return (elem == last()) ? 0 : elem->next();
145  }
146 
147  /// returns predecessor of \p elem or 0, if \p elem is the first element.
148  T* prev(const T *elem) const
149  {
150  return (elem == first()) ? 0 : elem->prev();
151  }
152  //@}
153 
154 
155  //---------------------------------------
156  /**@name Extension */
157  //@{
158  /// appends \p elem to end of list.
159  void append (T* elem)
160  {
161  if (last())
162  {
163  last()->next() = elem;
164  elem->prev() = last();
165  }
166  else
167  this->the_first = elem;
168  this->the_last = elem;
169  }
170 
171  /// prepends \p elem at beginnig of list.
172  void prepend(T* elem)
173  {
174  if (first())
175  {
176  elem->next() = first();
177  first()->prev() = elem;
178  }
179  else
180  this->the_last = elem;
181  this->the_first = elem;
182  }
183 
184  /// inserts \p elem after \p after.
185  void insert (T* elem, T* after)
186  {
187  assert(find(after));
188  if (after == last())
189  append(elem);
190  else
191  {
192  elem->next() = after->next();
193  elem->prev() = after;
194  after->next() = elem->next()->prev() = elem;
195  }
196  }
197 
198  /// appends \p list to end of list.
199  void append (IdList<T>& list)
200  {
201  if (list.first())
202  {
203  append(list.first());
204  this->the_last = list.last();
205  }
206  }
207 
208  /// prepends \p list at beginnig of list.
209  void prepend(IdList<T>& list)
210  {
211  if (list.first())
212  {
213  prepend(list.last());
214  this->the_first = list.the_first;
215  }
216  }
217 
218  /// inserts \p list after \p after.
219  void insert (IdList<T>& list, T* after)
220  {
221  assert(find(after));
222  if (list.first())
223  {
224  list.last()->next() = after->next();
225  list.first()->prev() = after;
226  after->next() = list.first();
227  if (after == last())
228  this->the_last = list.last();
229  else
230  list.last()->next()->prev() = list.last();
231  }
232  }
233  //@}
234 
235  //---------------------------------------
236  /**@name Removal */
237  //@{
238  /// removes element following \p after.
239  void remove_next(T* after)
240  {
241  remove(next(after));
242  }
243 
244  /// removes \p elem from list.
245  void remove(T* elem)
246  {
247  if (elem == first())
248  {
249  this->the_first = next(elem);
250  if (first() == 0)
251  this->the_last = 0;
252  }
253  else if (elem == last())
254  this->the_last = elem->prev();
255  else
256  {
257  elem->next()->prev() = elem->prev();
258  elem->prev()->next() = elem->next();
259  }
260  }
261 
262  /// removes sublist \p list.
263  void remove(IdList<T>& list)
264  {
265  if (first() != 0 && list.first() != 0)
266  {
267  assert(find(list.first()));
268  assert(find(list.last()));
269  if (first() == list.first())
270  {
271  if (last() == list.last())
272  this->the_first = this->the_last = 0;
273  else
274  this->the_first = list.last()->next();
275  }
276  else if (last() == list.last())
277  this->the_last = list.last()->prev();
278  else
279  {
280  T* after = first();
281  for (; after->next() != list.first(); after = after->next())
282  ;
283  if (last() == list.last())
284  this->the_last = after;
285  else
286  after->next() = list.last()->next();
287  }
288  }
289  }
290  //@}
291 
292  //---------------------------------------
293  /**@name Miscellaneous */
294  //@{
295  /// adjusts list pointers to a new memory address.
296  /** When all elements have been moved in memory (e.g. because of
297  reallocation) with a fixed offset \p delta, the list will be reset
298  to the new adresses.
299  */
300  void move(ptrdiff_t delta)
301  {
302  if (this->the_first)
303  {
304  T* elem;
305  IsList<T>::move(delta);
306  for (elem = last(); elem; elem = prev(elem))
307  if (elem != first())
308  elem->prev() = reinterpret_cast<T*>(
309  reinterpret_cast<char*>(elem->prev()) + delta);
310  }
311  }
312 
313  /// consistency check.
314  bool isConsistent() const
315  {
316 #ifdef ENABLE_CONSISTENCY_CHECKS
317  const T* my_first = first();
318  const T* my_last = last();
319  for (const T * it = my_first; it; it = next(it))
320  {
321  if (it != my_first && it->prev()->next() != it)
322  return MSGinconsistent("IdList");
323 
324  if (it != my_last && it->next()->prev() != it)
325  return MSGinconsistent("IdList");
326  }
327  return IsList<T>::isConsistent();
328 #else
329  return true;
330 #endif
331  }
332  //@}
333 
334  //---------------------------------------
335  /**@name Constructors / Destructors */
336  //@{
337  /// default constructor.
338  /** The default constructor may also be used to construct a sublist, by
339  providing a \p first and a \p last element. Element \p last must be a
340  successor of \p first.
341  */
342  explicit
343  IdList(T* pfirst = 0, T* plast = 0)
344  : IsList<T>(pfirst, plast)
345  {
346  assert(isConsistent());
347  }
348 
349  //@}
350 };
351 
352 } // namespace soplex
353 #endif // _IDLIST_H_
354 
355 //-----------------------------------------------------------------------------
356 //Emacs Local Variables:
357 //Emacs mode:c++
358 //Emacs c-basic-offset:3
359 //Emacs tab-width:8
360 //Emacs indent-tabs-mode:nil
361 //Emacs End:
362 //-----------------------------------------------------------------------------