qxLib
sort.inl
Go to the documentation of this file.
1 /**
2 
3  @file sort.inl
4  @author Khrapov
5  @date 29.04.2023
6  @copyright © Nick Khrapov, 2023. All right reserved.
7 
8 **/
9 
10 namespace qx
11 {
12 
13 template<class random_it_t>
14 inline bool sort_required(random_it_t begin, random_it_t end)
15 {
16  if (end - begin < 2)
17  return false;
18  else
19  return true;
20 }
21 
22 template<class random_it_t, class compare_t>
23 void sort_insertion(random_it_t begin, random_it_t end, compare_t compare)
24 {
25  if (!sort_required(begin, end))
26  return;
27 
28  for (random_it_t i = begin + 1; i != end; ++i) //-V756
29  for (random_it_t j = i; j != begin && compare(*j, *(j - 1)); --j)
30  std::iter_swap(j - 1, j);
31 }
32 
33 template<class container_t, class compare_t>
34 void sort_insertion(container_t& cont, compare_t compare)
35 {
36  sort_insertion(cont.begin(), cont.end(), compare);
37 }
38 
39 template<class random_it_t, class compare_t>
40 void sort_selection(random_it_t begin, random_it_t end, compare_t compare)
41 {
42  if (!sort_required(begin, end))
43  return;
44 
45  for (random_it_t it = begin; it != end; ++it)
46  {
47  random_it_t itMin = it;
48  for (random_it_t j = it + 1; j != end; ++j)
49  if (compare(*j, *itMin))
50  itMin = j;
51 
52  std::iter_swap(it, itMin);
53  }
54 }
55 
56 template<class container_t, class compare_t>
57 void sort_selection(container_t& cont, compare_t compare)
58 {
59  sort_selection(cont.begin(), cont.end(), compare);
60 }
61 
62 template<class random_it_t, class compare_t>
63 void sort_bubble(random_it_t begin, random_it_t end, compare_t compare)
64 {
65  if (!sort_required(begin, end))
66  return;
67 
68  bool bSorted = false;
69  while (!bSorted)
70  {
71  bSorted = true;
72  for (random_it_t it = begin, itEnd = end - 1; it != itEnd; ++it)
73  {
74  if (compare(*(it + 1), *it))
75  {
76  std::iter_swap(it, it + 1);
77  bSorted = false;
78  }
79  }
80  }
81 }
82 
83 template<class container_t, class compare_t>
84 void sort_bubble(container_t& cont, compare_t compare)
85 {
86  sort_bubble(cont.begin(), cont.end(), compare);
87 }
88 
89 template<class random_it_t, class compare_t>
90 void adjust_heap(random_it_t begin, size_t nHeapSize, size_t nPosition, compare_t compare)
91 {
92  using iter_diff = decltype(random_it_t() - random_it_t());
93 
94  while (nPosition < nHeapSize)
95  {
96  size_t nChildPos = nPosition * 2 + 1;
97  if (nChildPos < nHeapSize)
98  {
99  if ((nChildPos + 1 < nHeapSize)
100  && compare(
101  *(begin + static_cast<iter_diff>(nChildPos)),
102  *(begin + static_cast<iter_diff>(nChildPos) + 1)))
103  {
104  nChildPos += 1;
105  }
106 
107  if (compare(*(begin + static_cast<iter_diff>(nChildPos)), *(begin + static_cast<iter_diff>(nPosition))))
108  {
109  return;
110  }
111  else
112  {
113  std::iter_swap(begin + static_cast<iter_diff>(nPosition), begin + static_cast<iter_diff>(nChildPos));
114  }
115  }
116  nPosition = nChildPos;
117  }
118 }
119 
120 template<class random_it_t, class compare_t>
121 void make_heap(random_it_t begin, random_it_t end, compare_t compare)
122 {
123  auto max = end - begin;
124  for (int i = static_cast<int>(max) / 2; i >= 0; --i)
125  {
126  adjust_heap(
127  begin,
128  static_cast<size_t>(max),
129  static_cast<size_t>(i), //-V201
130  compare);
131  }
132 }
133 
134 template<class random_it_t, class compare_t>
135 void sort_heap(random_it_t begin, random_it_t end, compare_t compare)
136 {
137  if (!sort_required(begin, end))
138  return;
139 
140  qx::make_heap(begin, end, compare);
141 
142  auto nLastPosition = end - begin - 1;
143  while (nLastPosition > 0)
144  {
145  std::iter_swap(begin, begin + nLastPosition);
146  adjust_heap(begin, static_cast<size_t>(nLastPosition), 0u, compare);
147  --nLastPosition;
148  }
149 }
150 
151 template<class container_t, class compare_t>
152 void sort_heap(container_t& cont, compare_t compare)
153 {
154  qx::sort_heap(cont.begin(), cont.end(), compare);
155 }
156 
157 template<class random_it_t, class compare_t>
158 void sort_quick_hoare(random_it_t begin, random_it_t end, compare_t compare)
159 {
160  if (!sort_required(begin, end))
161  return;
162 
163  i64 nRight = (end - begin) - 1;
164  i64 nLeft = 0;
165 
166  if (nRight > nLeft)
167  {
168  auto nPivot = *(begin + nLeft + (nRight - nLeft) / 2);
169 
170  while (nLeft <= nRight)
171  {
172  while (compare(*(begin + nLeft), nPivot))
173  ++nLeft;
174 
175  while (compare(nPivot, *(begin + nRight)))
176  --nRight;
177 
178  if (nLeft <= nRight)
179  std::iter_swap(begin + nLeft++, begin + nRight--);
180  }
181 
182  sort_quick_hoare(begin, begin + (nRight + 1), compare);
183  sort_quick_hoare(begin + nLeft, end, compare);
184  }
185 }
186 
187 template<class container_t, class compare_t>
188 void sort_quick_hoare(container_t& cont, compare_t compare)
189 {
190  sort_quick_hoare(cont.begin(), cont.end(), compare);
191 }
192 
193 template<class random_it_t, class compare_t>
194 void sort_quick_three_way(random_it_t begin, random_it_t end, compare_t compare)
195 {
196  if (!sort_required(begin, end))
197  return;
198 
199  const i64 nRight = (end - begin) - 1;
200  i64 nLeft = 0;
201 
202  if (nRight > 0)
203  {
204  auto pivot = *(begin + nLeft + (nRight - nLeft) / 2);
205 
206  i64 i = nLeft;
207  i64 j = nRight;
208 
209  i64 nIndex1 = nLeft - 1;
210  i64 nIndex2 = nRight + 1;
211 
212  while (i <= j)
213  {
214  while (compare(*(begin + i), pivot))
215  ++i;
216 
217  while (compare(pivot, *(begin + j)))
218  --j;
219 
220  if (i < j)
221  {
222  if (compare(pivot, *(begin + i)))
223  {
224  if (compare(*(begin + j), pivot))
225  {
226  // nLeft[i] != nPivot and nLeft[j] != nPivot
227  std::swap(*(begin + i), *(begin + j));
228  }
229  else
230  {
231  // nLeft[i] != nPivot and nLeft[j] == nPivot
232  std::swap(*(begin + i), *(begin + j));
233  std::swap(*(begin + ++nIndex1), *(begin + i));
234  }
235  }
236  else if (compare(*(begin + j), pivot))
237  {
238  // nLeft[i] == nPivot nLeft[j] != nPivot
239  std::iter_swap(begin + i, begin + j);
240  std::iter_swap(begin + j, begin + --nIndex2);
241  }
242  else
243  {
244  // nLeft[i] == nPivot and nLeft[j] == nPivot
245  std::iter_swap(begin + ++nIndex1, begin + i);
246  std::iter_swap(begin + j, begin + --nIndex2);
247  }
248 
249  ++i;
250  --j;
251  }
252  else if (i == j)
253  {
254  j = i - 1;
255  ++i;
256  break;
257  }
258  }
259 
260  for (i64 k = nLeft; k <= nIndex1; ++k)
261  std::iter_swap(begin + k, begin + j--);
262 
263  for (i64 k = nRight; k >= nIndex2; --k)
264  std::iter_swap(begin + i++, begin + k);
265 
266  sort_quick_three_way(begin, begin + (j + 1), compare);
267  sort_quick_three_way(begin + i, end, compare);
268  }
269 }
270 
271 template<class container_t, class compare_t>
272 void sort_quick_three_way(container_t& cont, compare_t compare)
273 {
274  sort_quick_three_way(cont.begin(), cont.end(), compare);
275 }
276 
277 template<class random_it_t, class compare_t>
278 void sort_quick_dual_pivot(random_it_t begin, random_it_t end, compare_t compare)
279 {
280  if (!sort_required(begin, end))
281  return;
282 
283  i64 nRight = (end - begin) - 1;
284  i64 nLeft = 0;
285 
286  if (nRight > nLeft)
287  {
288  std::iter_swap(begin + nLeft + (nRight - nLeft) / 3, begin + nLeft);
289  std::iter_swap(begin + nRight - (nRight - nLeft) / 3, begin + nRight);
290 
291  if (compare(*(begin + nRight), *(begin + nLeft)))
292  std::iter_swap(begin + nLeft, begin + nRight);
293 
294  if (nRight - nLeft == 1)
295  return;
296 
297  auto pivot1 = *(begin + nLeft);
298  auto pivot2 = *(begin + nRight);
299 
300  if (compare(pivot1, pivot2))
301  {
302  i64 nLess = nLeft + 1;
303  i64 nGreater = nRight - 1;
304 
305  while (!compare(*(begin + nGreater), pivot2))
306  --nGreater;
307 
308  while (!compare(pivot1, *(begin + nLess)))
309  ++nLess;
310 
311  i64 k = nLess;
312  while (k <= nGreater)
313  {
314  if (!compare(pivot1, *(begin + k)))
315  {
316  std::iter_swap(begin + k, begin + nLess++);
317  }
318  else if (!compare(*(begin + k), pivot2))
319  {
320  std::iter_swap(begin + k, begin + nGreater--);
321  while (!compare(*(begin + nGreater), pivot2))
322  --nGreater;
323 
324  if (!compare(pivot1, *(begin + k)))
325  std::iter_swap(begin + k, begin + nLess++);
326  }
327 
328  k++;
329  }
330 
331  std::iter_swap(begin + nLess - 1, begin + nLeft);
332  std::iter_swap(begin + nGreater + 1, begin + nRight);
333 
334  sort_quick_dual_pivot(begin, begin + nLess - 1, compare);
335  sort_quick_dual_pivot(begin + nGreater + 2, end, compare);
336  sort_quick_dual_pivot(begin + nLess, begin + nGreater + 1, compare);
337  }
338  else
339  {
340  // pivot1 == pivot2
341  i64 nLess = nLeft + 1;
342  i64 nGreater = nRight - 1;
343 
344  while (compare(pivot1, *(begin + nGreater)))
345  --nGreater;
346 
347  while (compare(*(begin + nLess), pivot1))
348  ++nLess;
349 
350  i64 k = nLess;
351  while (k <= nGreater)
352  {
353  if (compare(*(begin + k), pivot1))
354  {
355  std::iter_swap(begin + k, begin + nLess++);
356  }
357  else if (compare(pivot1, *(begin + k)))
358  {
359  std::iter_swap(begin + k, begin + nGreater--);
360  while (compare(pivot1, *(begin + nGreater)))
361  --nGreater;
362 
363  if (compare(*(begin + k), pivot1))
364  std::iter_swap(begin + k, begin + nLess++);
365  }
366 
367  k++;
368  }
369 
370  std::iter_swap(begin + nLess - 1, begin + nLeft);
371  std::iter_swap(begin + nGreater + 1, begin + nRight);
372 
373  sort_quick_dual_pivot(begin, begin + nLess - 1, compare);
374  sort_quick_dual_pivot(begin + nGreater + 2, end, compare);
375  }
376  }
377 }
378 
379 template<class container_t, class compare_t>
380 void sort_quick_dual_pivot(container_t& cont, compare_t compare)
381 {
382  sort_quick_dual_pivot(cont.begin(), cont.end(), compare);
383 }
384 
385 template<class random_it_t, class compare_t>
386 void merge(random_it_t begin, random_it_t end, compare_t compare, vector_of_values<random_it_t>* pPreallocatedBuffer)
387 {
388  using iter_diff = decltype(random_it_t() - random_it_t());
389 
390  u64 nInd1 = 0;
391  u64 nInd2 = 0;
392  random_it_t mid = begin + (end - begin) / 2;
393 
394  const bool bCreateBuffer = !pPreallocatedBuffer;
395  if (bCreateBuffer)
396  pPreallocatedBuffer = new vector_of_values<random_it_t>;
397 
398  pPreallocatedBuffer->resize(static_cast<size_t>(end - begin));
399 
400  // merge
401  while ((begin + static_cast<iter_diff>(nInd1) < mid) && (mid + static_cast<iter_diff>(nInd2) < end))
402  {
403  if (compare(*(begin + static_cast<iter_diff>(nInd1)), *(mid + static_cast<iter_diff>(nInd2))))
404  {
405  (*pPreallocatedBuffer)[nInd1 + nInd2] = *(begin + static_cast<iter_diff>(nInd1));
406 
407  nInd1++;
408  }
409  else
410  {
411  (*pPreallocatedBuffer)[nInd1 + nInd2] = *(mid + static_cast<iter_diff>(nInd2));
412 
413  nInd2++;
414  }
415  }
416 
417  // append tails
418  while (begin + static_cast<iter_diff>(nInd1) < mid)
419  {
420  (*pPreallocatedBuffer)[nInd1 + nInd2] = *(begin + static_cast<iter_diff>(nInd1));
421 
422  nInd1++;
423  }
424 
425  while (mid + static_cast<iter_diff>(nInd2) < end)
426  {
427  (*pPreallocatedBuffer)[nInd1 + nInd2] = *(mid + static_cast<iter_diff>(nInd2));
428 
429  nInd2++;
430  }
431 
432  // copy to source
433  for (auto it = begin; it < end; ++it)
434  *it = (*pPreallocatedBuffer)[static_cast<size_t>(it - begin)];
435 
436  if (bCreateBuffer)
437  delete pPreallocatedBuffer;
438 }
439 
440 template<class random_it_t, class compare_t>
442  random_it_t begin,
443  random_it_t end,
444  compare_t compare,
445  vector_of_values<random_it_t>* pPreallocatedBuffer)
446 {
447  if (!sort_required(begin, end))
448  return;
449 
450  const bool bCreateBuffer = !pPreallocatedBuffer;
451  if (bCreateBuffer)
452  pPreallocatedBuffer = new vector_of_values<random_it_t>;
453 
454  // preallocate max required size and use same vector
455  // has no effect in reqursive calls
456  auto nLen = end - begin;
457  pPreallocatedBuffer->reserve(static_cast<size_t>(nLen));
458 
459  sort_merge(begin, begin + nLen / 2, compare, pPreallocatedBuffer);
460  sort_merge(begin + nLen / 2, end, compare, pPreallocatedBuffer);
461  merge(begin, end, compare, pPreallocatedBuffer);
462 
463  if (bCreateBuffer)
464  delete pPreallocatedBuffer;
465 }
466 
467 template<class container_t, class compare_t>
469  container_t& cont,
470  compare_t compare,
471  vector_of_values<typename container_t::iterator>* pPreallocatedBuffer)
472 {
473  sort_merge(cont.begin(), cont.end(), compare, pPreallocatedBuffer);
474 }
475 
476 template<class random_it_t, class compare_t>
477 bool sort_counting(random_it_t begin, random_it_t end, compare_t compare, size_t nMaxBufferSize)
478 {
479  static_assert(std::is_integral_v<iterator_value_t<random_it_t>>, "Integral type required for counting sort");
480 
481  if (end - begin < 2)
482  return true;
483 
484  auto minMax = std::minmax_element(begin, end);
485  auto min = *minMax.first;
486  auto max = *minMax.second;
487 
488  const size_t nSizeRequired = static_cast<size_t>(max) - min + 1;
489  if (nSizeRequired <= nMaxBufferSize)
490  {
491  std::vector<size_t> counts(nSizeRequired, 0);
492 
493  for (random_it_t it = begin; it < end; ++it)
494  ++counts[static_cast<size_t>(*it) - min];
495 
496  if constexpr (compare(0, 1))
497  {
498  // nLess
499  for (size_t i = 0; i < nSizeRequired; ++i)
500  begin = std::fill_n(begin, counts[i], min++);
501  }
502  else
503  {
504  // nGreater
505  for (size_t i = 0; i < nSizeRequired; ++i)
506  begin = std::fill_n(begin, counts[nSizeRequired - i - 1], max--);
507  }
508 
509  return true;
510  }
511  else
512  return false;
513 }
514 
515 template<class container_t, class compare_t>
516 bool sort_counting(container_t& cont, compare_t compare, size_t nMaxBufferSize)
517 {
518  return sort_counting(cont.begin(), cont.end(), compare, nMaxBufferSize);
519 }
520 
521 template<class random_it_t, class compare_t>
522 void sort(random_it_t begin, random_it_t end, compare_t compare)
523 {
524  bool bSorted = false;
525 
526  // if sort_counting is available
527  if constexpr (std::is_integral_v<iterator_value_t<random_it_t>>)
528  {
529  // with small container size quick is faster
530  if (end - begin > 1000)
531  bSorted = sort_counting(begin, end, compare);
532  }
533 
534  // in some cases with small container size std::sort() may be faster,
535  // but sort_quick_dual_pivot() faster in general, especially with big sizes
536  if (!bSorted)
537  sort_quick_dual_pivot(begin, end, compare);
538 }
539 
540 template<class container_t, class compare_t>
541 void sort(container_t& cont, compare_t compare)
542 {
543  qx::sort(cont.begin(), cont.end(), compare);
544 }
545 
546 } // namespace qx
void sort_selection(random_it_t begin, random_it_t end, compare_t compare=compare_t())
Selection sort.
Definition: sort.inl:40
void sort_quick_three_way(random_it_t begin, random_it_t end, compare_t compare=compare_t())
Quick sort based on three-way partitioning.
Definition: sort.inl:194
void sort_insertion(random_it_t begin, random_it_t end, compare_t compare=compare_t())
Insertion sort.
Definition: sort.inl:23
void sort_heap(random_it_t begin, random_it_t end, compare_t compare=compare_t())
Heap sort.
Definition: sort.inl:135
void sort_bubble(random_it_t begin, random_it_t end, compare_t compare=compare_t())
Bubble sort.
Definition: sort.inl:63
void sort(random_it_t begin, random_it_t end, compare_t compare=compare_t())
Sort by the most suitable algorithm.
Definition: sort.inl:522
bool sort_counting(random_it_t begin, random_it_t end, compare_t compare=compare_t(), size_t nMaxBufferSize=SORT_COUNTING_MAX_BUFFER_SIZE)
Counting sort for integral values.
Definition: sort.inl:477
void adjust_heap(random_it_t begin, size_t nHeapSize, size_t nPosition, compare_t compare=compare_t())
Adjust heap.
Definition: sort.inl:90
bool sort_required(random_it_t begin, random_it_t end)
Check if sort really needed.
Definition: sort.inl:14
void sort_quick_hoare(random_it_t begin, random_it_t end, compare_t compare=compare_t())
Quick sort based on Hoare partitioning.
Definition: sort.inl:158
void make_heap(random_it_t begin, random_it_t end, compare_t compare=compare_t())
Make heap at range.
Definition: sort.inl:121
void merge(random_it_t begin, random_it_t end, compare_t compare=compare_t(), vector_of_values< random_it_t > *pPreallocatedBuffer=nullptr)
Merge with temp vector.
Definition: sort.inl:386
void sort_quick_dual_pivot(random_it_t begin, random_it_t end, compare_t compare=compare_t())
Quick sort based on dual-pivot partitioning.
Definition: sort.inl:278
void sort_merge(random_it_t begin, random_it_t end, compare_t compare=compare_t(), vector_of_values< random_it_t > *pPreallocatedBuffer=nullptr)
Merge sort based on merge with temp vector.
Definition: sort.inl:441
uint64_t u64
Definition: typedefs.h:27