SoPlex Doxygen Documentation
islist.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 islist.h
17  * @brief Generic single linked list.
18  */
19 #ifndef _ISLIST_H_
20 #define _ISLIST_H_
21 
22 #include <assert.h>
23 #include <iostream>
24 
25 
26 namespace soplex
27 {
28 
29 //---------------------------------------------------------------------
30 // class IsElement<T>
31 //---------------------------------------------------------------------
32 
33 /**@brief Elements for \ref soplex::IsList "IsList"s.
34  @ingroup Elementary
35 
36  Class IsElement allows to easily construct list elements for an intrusive
37  single linked list IsList out of a template class T. It adds a next
38  pointer to each element. An instance of IdElement<T> a can be used just
39  like an instance of T itself, except that method next() has been
40  added (thereby overriding any method next() defined in T).
41  */
42 template < class T >
43 class IsElement : public T
44 {
45 protected:
46 
47  //--------------------------
48  /**@name Data */
49  //@{
50  IsElement<T>* the_next; ///< pointer to next element in the IsList.
51  //@}
52 
53 public:
54 
55  //---------------------------------
56  /**@name Successor */
57  //@{
58  ///
60  {
61  return the_next;
62  }
63  /// returns the next element in the IsList.
64  IsElement<T>* next() const
65  {
66  return the_next;
67  }
68  //@}
69 
70  //------------------------------------
71  /**@name Constructors / destructors */
72  //@{
73  /// default constructor.
75  {}
76  ///
77  explicit
78  IsElement(const T& old)
79  : T(old)
80  , the_next(0)
81  {}
82  /// copy constructor.
83  /** Only the element itself is copied, while the link to the next list
84  element is set to a zero pointer.
85  */
86  IsElement(const IsElement<T>& old)
87  : T(old)
88  , the_next(0)
89  {}
90 
91 };
92 
93 //---------------------------------------------------------------------
94 // class IsList<T>
95 //---------------------------------------------------------------------
96 
97 /**@brief Generic single linked list.
98  @ingroup Elementary
99 
100  Class IsList implements an intrusive single linked list of elements of a
101  template class T. As an \em intrusive list, the objects of type T
102  must provide methods next() for setting and inquiring a pointer to the
103  next element in a list. The user is responsible for not modifying the
104  next() pointer of elements currently residing in a list, which may destroy
105  the lists integrity. For this, class IsList provides enough methods for
106  modifying a list in a save way. See the method list for a description.
107  */
108 template < class T >
109 class IsList
110 {
111 protected:
112  T* the_first; ///< the first element in the IsList.
113  T* the_last; ///< the last element in the IsList.
114 
115 public:
117 
118  //--------------------------
119  /**@name Extension */
120  //@{
121  /// appends \p elem to IsList.
122  void append(T* elem)
123  {
124  if (the_last)
125  the_last->next() = elem;
126  else
127  the_first = elem;
128  the_last = elem;
129  }
130 
131  /// prepends \p elem to IsList.
132  void prepend(T* elem)
133  {
134  if (the_first)
135  elem->next() = the_first;
136  else
137  the_last = elem;
138  the_first = elem;
139  }
140 
141  /// inserts \p elem to IsList after its element \p after.
142  void insert(T* elem, T* after)
143  {
144  assert(find(after));
145  if (after == the_last)
146  append(elem);
147  else
148  {
149  elem->next() = after->next();
150  after->next() = elem;
151  }
152  }
153 
154  /// appends all elements of \p list to IsList.
155  /** Appending one list to another keeps the appended \p list. Instead,
156  \p list remains an own IsList which is then part of the
157  concatenated list. This means that modifying \p list will modify the
158  concateneted list as well and vice versa. The programmer is
159  responsible for such changes not to yield inconsistent lists.
160  */
161  void append(IsList<T>& list)
162  {
163  if (list.the_first)
164  {
165  append(list.the_first);
166  the_last = list.the_last;
167  }
168  }
169 
170  /// prepends all elements of \p list to IsList.
171  /** Appending one list to another keeps the appended \p list. Instead,
172  \p list remains an own IsList which is then part of the
173  concatenated list. This means that modifying \p list will modify the
174  concateneted list as well and vice versa. The programmer is
175  responsible for such changes not to yield inconsistent lists.
176  */
177  void prepend(IsList<T>& list)
178  {
179  if (list.the_first)
180  {
181  prepend(list.the_last);
182  the_first = list.the_first;
183  }
184  }
185 
186  /// inserts all elements of \p list after element \p after of an IsList.
187  /** Inserting one list into another keeps the appended \p list. Instead,
188  \p list remains an own IsList which is then part of the
189  concatenated list. This means that modifying \p list will modify the
190  concateneted list as well and vice versa. The programmer is
191  responsible for such changes not to yield inconsistent lists.
192  */
193  void insert(IsList<T>& list, T*after)
194  {
195  assert(find(after));
196  if (list.the_first)
197  {
198  list.the_last->next() = after->next();
199  after->next() = list.first();
200  if (after == last())
201  the_last = list.last();
202  }
203  }
204  //@}
205 
206  //--------------------------
207  /**@name Removal */
208  //@{
209  /// removes the successor of \p after from an IsList.
210  void remove_next(T *after)
211  {
212  assert(find(after));
213  if (after->next())
214  {
215  if (after->next() == last())
216  the_last = after;
217  after->next() = after->next()->next();
218  }
219  }
220 
221  /// removes element \p elem from an IsList.
222  void remove(const T *elem)
223  {
224  if (the_first)
225  {
226  if (elem == the_first)
227  {
228  the_first = next(elem);
229  if (the_first == 0)
230  the_last = 0;
231  }
232  else
233  {
234  T *after = the_first;
235  for (; after != the_last; after = after->next())
236  if (after->next() == elem)
237  {
238  remove_next(after);
239  return;
240  }
241  }
242  }
243  }
244 
245  /// removes all elements of \p list from an IsList.
246  /** Removing \p list from an IsList requires \p list to be part of the
247  IsList. Such a situation can be achieved by previously adding
248  (i.e., #append%ing, #insert%ing or #prepend%ing) a list or
249  explicitely constructing a sublist with method sublist().
250  */
251  void remove(IsList<T>& list)
252  {
253  if (the_first != 0 && list.the_first != 0)
254  {
255  assert(find(list.first()));
256  assert(find(list.last()));
257  if (the_first == list.the_first)
258  {
259  if (the_last == list.the_last)
260  the_first = the_last = 0;
261  else
262  the_first = list.the_last->next();
263  }
264  else
265  {
266  T *after = the_first;
267  for (; after->next() != list.the_first; after = after->next())
268  ;
269  if (the_last == list.the_last)
270  the_last = after;
271  else
272  after->next() = list.the_last->next();
273  }
274  }
275  }
276 
277  /// removes all elements from an IsList.
278  void clear()
279  {
280  the_first = the_last = 0;
281  }
282  //@}
283 
284  //--------------------------
285  /**@name Access */
286  //@{
287  /// returns the IsList's first element.
288  T* first() const
289  {
290  return the_first;
291  }
292 
293  /// returns the IsList's last element.
294  T* last() const
295  {
296  return the_last;
297  }
298 
299  /// returns successor of \p elem in an IsList.
300  /** The successor of \p elem in a list generally corresponds to the
301  element returned by elem->next(). However, if \p elem is the last
302  element in an IsList, this method will return 0, whereas
303  elem->next() may yield an arbitrary value. For example, if the
304  current list is actually a sublist of another, larger IsList,
305  elem->next() returns the successor of \p elem in this larger
306  IsList.
307  */
308  T* next(const T *elem) const
309  {
310  return (elem == the_last) ? 0 : elem->next();
311  }
312 
313  /// returns the number of elements in IsList.
314  int length() const
315  {
316  int num;
317  if (the_first)
318  {
319  T *test = the_first;
320  for (num = 1; test != the_last; test = test->next())
321  ++num;
322  return num;
323  }
324  return 0;
325  }
326 
327  /// returns the position of element \p elem within IsList.
328  int find(const T* elem) const
329  {
330  const T* test;
331 
332  assert(elem != 0);
333 
334  for(test = the_first; test != 0; test = next(test))
335  if (test == elem)
336  return 1;
337 
338  return 0;
339  }
340 
341  /// constructs sublist of an IsList.
342  /** Returns a new IsList containing a sublist of an IsList starting
343  with element \p start and reaching up to element \p end. Both must be
344  members of the IsList or 0, in which case the first and last
345  element are used, respectively.
346  */
347  IsList<T>sublist(const T* start = 0, const T* end = 0) const
348  {
349  IsList<T>part = *this;
350  if (start)
351  {
352  assert(find(start));
353  part.the_first = const_cast<T*>(start);
354  }
355  if (end)
356  {
357  assert(part.find(end));
358  part.the_last = const_cast<T*>(end);
359  }
360  return part;
361  }
362  //@}
363 
364  //--------------------------
365  /**@name Miscellaneous */
366  //@{
367  /// adjusts list pointers to a new memory address.
368  /** This method is of a rather technical nature. If all list elements
369  are taken form one array of elements, in certain circumstances the
370  user may be forced to realloc this array. As a consequence all
371  next() pointers of the list elements would become invalid.
372  However, all addresses will be changed by a constant offset \p delta.
373  Then move( \p delta ) may be called, which adjusts the next()
374  pointers of all elements in the list.
375  */
376  void move(ptrdiff_t delta)
377  {
378  if (the_first)
379  {
380  T* elem;
381  the_last = reinterpret_cast<T*>(reinterpret_cast<char*>(the_last) + delta);
382  the_first = reinterpret_cast<T*>(reinterpret_cast<char*>(the_first) + delta);
383  for (elem = first(); elem; elem = next(elem))
384  if (elem != last())
385  elem->next() = reinterpret_cast<T*>(reinterpret_cast<char*>(elem->next()) + delta);
386  }
387  }
388 
389  /// consistency check.
390  bool isConsistent() const
391  {
392 #ifdef ENABLE_CONSISTENCY_CHECKS
393  if (first() != 0 && last() == 0)
394  return MSGinconsistent("IsList");
395 
396  if (first() == 0 && last() != 0)
397  return MSGinconsistent("IsList");
398 
399  if (first() && find(last()) == 0)
400  return MSGinconsistent("IsList");
401 #endif
402 
403  return true;
404  }
405  //@}
406 
407  //------------------------------------
408  /**@name Constructors / Destructors */
409  //@{
410  /// default constructor.
411  /** The default constructor may be used to setup a (sub-)list, by
412  specifying a \p first and \p last element. Then \p last must be a
413  successor of \p first.
414  */
415  explicit
416  IsList(T* pfirst = 0, T* plast = 0)
417  : the_first(pfirst), the_last(plast)
418  {
419  if (pfirst)
420  {
421  assert(plast != 0);
422  assert(find(plast));
423  }
424 
425  assert(isConsistent());
426  }
427  //@}
428 };
429 
430 } // namespace soplex
431 #endif // _ISLIST_H_
432 
433 //-----------------------------------------------------------------------------
434 //Emacs Local Variables:
435 //Emacs mode:c++
436 //Emacs c-basic-offset:3
437 //Emacs tab-width:8
438 //Emacs indent-tabs-mode:nil
439 //Emacs End:
440 //-----------------------------------------------------------------------------