SoPlex Doxygen Documentation
sorter.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 
17 /**@file sorter.h
18  * @brief Generic QuickSort implementation.
19  */
20 #ifndef _SORTER_H_
21 #define _SORTER_H_
22 
23 #include <assert.h>
24 
25 namespace soplex
26 {
27 /// Generic QuickSort implementation.
28 /** This template function sorts an array \p t holding \p n elements of
29  type T using \p compare for comparisons. Class COMPARATOR must provide
30  an overloaded operator()(const T& t1,const T& t2) which returns
31  - < 0, if \p t1 is to appear before \p t2,
32  - = 0, if \p t1 and \p t2 can appear in any order, or
33  - > 0, if \p t1 is to appear after \p t2.
34 */
35 
36 template < class T, class COMPARATOR >
37 static void sorter_qsort(T* t, int end, COMPARATOR& compare, int start = 0)
38 {
39  assert(start >= 0);
40 
41  /* nothing to sort for less than two elements */
42  if( end <= start + 1 )
43  return;
44 
45  int i0, i1, j;
46  Real c;
47 
48  T work, mid, tmp;
49 
50  work = t[start];
51  t[start] = t[(start + end) / 2];
52  t[(start + end) / 2] = work;
53 
54  mid = t[start];
55  work = t[end - 1];
56 
57  for (i0 = i1 = start, j = end - 1; i1 < j;)
58  {
59  c = compare(mid, work);
60  if (c > 0)
61  {
62  tmp = t[i0];
63  t[i0] = work;
64  i0++;
65  i1++;
66  work = t[i1];
67  t[i1] = tmp;
68  }
69  else if (c < 0)
70  {
71  t[j] = work;
72  --j;
73  work = t[j];
74  }
75  else
76  {
77  i1++;
78  tmp = t[i1];
79  t[i1] = work;
80  work = tmp;
81  }
82  }
83 
84  if (start < i0 - 1)
85  sorter_qsort(t, i0, compare, start);
86  if (i1 + 1 < end)
87  sorter_qsort(t, end, compare, i1 + 1);
88 }
89 
90 
91 /**@brief Generic implementation of Partial QuickSort.
92  *
93  * This template function sorts an array \p t holding \p n elements of
94  * type T partially using \p compare for comparisons, i.e. ensures that
95  * the \p size smallest elements are sorted to the front.
96  *
97  * Class COMPARATOR must provide an overloaded
98  * operator()(const T& t1,const T& t2) which returns
99  * - < 0, if \p t1 is to appear before \p t2,
100  * - = 0, if \p t1 and \p t2 can appear in any order, or
101  * - > 0, if \p t1 is to appear after \p t2.
102  */
103 template < class T, class COMPARATOR >
104 static int sorter_qsortPart(
105  T* t, /**< array of elements to be sorted between index start and end */
106  COMPARATOR& compare, /**< comparator */
107  int start, /**< index of first element in range to be sorted */
108  int end, /**< index of last element in range to be sorted, plus 1 */
109  int size, /**< guaranteed number of sorted elements */
110  int start2 = 0, /**< auxiliary start index of sub range used for recursive call */
111  int end2 = 0 /**< auxiliary end index of sub range used for recursive call */
112  )
113 {
114  assert(start >= 0);
115 
116  /* nothing to sort for less than two elements */
117  if( end < start + 1 )
118  return 0;
119  else if( end <= start + 1 )
120  return 1;
121 
122  T work;
123  T pivotval; /* value of pivot element */
124  T tmp;
125 
126  Real c;
127 
128  int pivotind; /* index of pivot element */
129  int i0; /* elements start, ..., i0-1 are strictly smaller than the pivot value */
130  int i1; /* elements i0, ..., i1-1 are equal to the pivot value */
131  int j; /* elements i1, ..., j-1 have arbitrary value; elements j, ..., end-1 are strictly greater than the pivot value */
132 
133  /* we assume that range {start, ..., start2-1} already contains the start2-start smallest elements in sorted order;
134  * start2 has to lie in {start, ..., end-1} */
135  if( start2 < start )
136  start2 = start;
137 
138  /* if all remaining elements should be sorted, we simply call standard quicksort */
139  if( start2+size >= end-1 )
140  {
141  sorter_qsort(t, end, compare, start2);
142  return end > end2 ? end-1 : end2-1;
143  }
144 
145  /* select middle element as pivot element */
146  pivotind = (start2+end)/2;
147  pivotval = t[pivotind];
148 
149  /* exchange first element with pivot element */
150  t[pivotind] = t[start2];
151  t[start2] = pivotval;
152 
153  /* the unsorted part covers {start2, ..., end-1} */
154  i0 = start2;
155  i1 = start2;
156  j = end-1;
157 
158  /* we start at the end of the unsorted range */
159  work = t[j];
160 
161  /* keep sorting as long as the part with arbitrary values is nonempty */
162  while( i1 < j )
163  {
164  /* compare work to pivot element */
165  c = compare(pivotval, work);
166 
167  /* work < mid */
168  if( c > 0 )
169  {
170  tmp = t[i0];
171  t[i0] = work;
172  i0++;
173  i1++;
174  work = t[i1];
175  t[i1] = tmp;
176  }
177  /* work > mid */
178  else if (c < 0)
179  {
180  t[j] = work;
181  j--;
182  work = t[j];
183  }
184  /* work == mid */
185  else
186  {
187  i1++;
188  tmp = t[i1];
189  t[i1] = work;
190  work = tmp;
191  }
192  }
193 
194  /* if we only need to sort less than half of the "<" part, use partial sort again */
195  if( 2*size <= i0 - start2 )
196  {
197  return sorter_qsortPart(t, compare, start, i0, size, start2, i1);
198  }
199  /* otherwise, and if we do not need to sort the ">" part, use standard quicksort on the "<" part */
200  else if( size <= i1 - start2 )
201  {
202  sorter_qsort(t, i0, compare, start2);
203  return i1-1;
204  }
205  /* otherwise we have to sort the "<" part fully (use standard quicksort) and the ">" part partially */
206  else
207  {
208  sorter_qsort(t, i0, compare, start);
209  return sorter_qsortPart(t, compare, start, end, size+start2-i1, i1+1, i1);
210  }
211 }
212 
213 } // namespace soplex
214 #endif // _SORTER_H_
215 
216 //-----------------------------------------------------------------------------
217 //Emacs Local Variables:
218 //Emacs mode:c++
219 //Emacs c-basic-offset:3
220 //Emacs tab-width:8
221 //Emacs indent-tabs-mode:nil
222 //Emacs End:
223 //-----------------------------------------------------------------------------