test_sample_sort.cpp 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. //----------------------------------------------------------------------------
  2. /// @file test_sample_sort.cpp
  3. /// @brief test sample_sort algorithm
  4. ///
  5. /// @author Copyright (c) 2016 Francisco Jose Tapia (fjtapia@gmail.com )\n
  6. /// Distributed under the Boost Software License, Version 1.0.\n
  7. /// ( See accompanying file LICENSE_1_0.txt or copy at
  8. /// http://www.boost.org/LICENSE_1_0.txt )
  9. /// @version 0.1
  10. ///
  11. /// @remarks
  12. //-----------------------------------------------------------------------------
  13. #include <ciso646>
  14. #include <cstdlib>
  15. #include <ctime>
  16. #include <algorithm>
  17. #include <vector>
  18. #include <random>
  19. #include <boost/sort/sample_sort/sample_sort.hpp>
  20. #include <boost/test/included/test_exec_monitor.hpp>
  21. #include <boost/test/test_tools.hpp>
  22. namespace bss = boost::sort;
  23. struct xk
  24. {
  25. unsigned tail : 4;
  26. unsigned num : 28;
  27. xk ( uint32_t n =0 , uint32_t t =0): tail (t), num(n){};
  28. bool operator< (xk A) const { return (num < A.num); };
  29. };
  30. void test1()
  31. {
  32. std::mt19937_64 my_rand(0);
  33. const uint32_t NMAX = 100000;
  34. std::vector<xk> V1, V2, V3;
  35. V1.reserve(NMAX);
  36. for (uint32_t i = 0; i < 8; ++i)
  37. {
  38. for (uint32_t k = 0; k < NMAX; ++k)
  39. { uint32_t NM = my_rand();
  40. xk G;
  41. G.num = NM >> 3;
  42. G.tail = i;
  43. V1.push_back(G);
  44. };
  45. };
  46. V3 = V2 = V1;
  47. bss::sample_sort(V1.begin(), V1.end());
  48. std::stable_sort(V2.begin(), V2.end());
  49. bss::sample_sort(V3.begin(), V3.end(), 0);
  50. BOOST_CHECK(V1.size() == V2.size());
  51. for (uint32_t i = 0; i < V1.size(); ++i)
  52. { BOOST_CHECK(V1[i].num == V2[i].num and V1[i].tail == V2[i].tail);
  53. };
  54. BOOST_CHECK(V3.size() == V2.size());
  55. for (uint32_t i = 0; i < V3.size(); ++i)
  56. { BOOST_CHECK(V3[i].num == V2[i].num and V3[i].tail == V2[i].tail);
  57. };
  58. };
  59. void test2(void)
  60. {
  61. const uint32_t NElem = 2000000;
  62. std::vector<uint64_t> V1;
  63. // ----------------- sorted elements ------------------------------------
  64. V1.clear();
  65. for (uint32_t i = 0; i < NElem; ++i) V1.push_back(i);
  66. bss::sample_sort(V1.begin(), V1.end());
  67. for (unsigned i = 1; i < NElem; i++)
  68. { BOOST_CHECK(V1[i - 1] <= V1[i]);
  69. };
  70. // ------------- reverse sorted elements --------------------------------
  71. V1.clear();
  72. for (uint32_t i = 0; i < NElem; ++i) V1.push_back(NElem - i);
  73. bss::sample_sort(V1.begin(), V1.end());
  74. for (unsigned i = 1; i < NElem; i++)
  75. { BOOST_CHECK(V1[i - 1] <= V1[i]);
  76. };
  77. // -------------------- equals elements -----------------------------------
  78. V1.clear();
  79. for (uint32_t i = 0; i < NElem; ++i) V1.push_back(1000);
  80. bss::sample_sort(V1.begin(), V1.end());
  81. for (unsigned i = 1; i < NElem; i++)
  82. { BOOST_CHECK(V1[i - 1] == V1[i]);
  83. };
  84. };
  85. void test3(void)
  86. {
  87. typedef std::less<uint64_t> compare;
  88. const uint32_t NElem = 2000000;
  89. std::vector<uint64_t> V1,V2,V3;
  90. std::mt19937_64 my_rand(0);
  91. for (uint32_t i = 0; i < NElem; ++i) V1.push_back(my_rand() % NElem);
  92. V3 = V2 = V1;
  93. std::stable_sort (V2.begin(), V2.end());
  94. // --------------- unsorted elements 0 threads ----------------------------
  95. V3 = V1;
  96. bss::sample_sort(V3.begin(), V3.end(), compare(), 0);
  97. for (unsigned i = 0; i < NElem; i++)
  98. { BOOST_CHECK(V3[i] == V2[i]);
  99. };
  100. // --------------- unsorted elements -------------------------------------
  101. V3 = V1;
  102. bss::sample_sort(V3.begin(), V3.end(), compare());
  103. for (unsigned i = 0; i < NElem; i++)
  104. { BOOST_CHECK(V3[i] == V2[i]);
  105. };
  106. // --------------- unsorted elements 100 threads ----------------------------
  107. V3 = V1;
  108. bss::sample_sort(V3.begin(), V3.end(), compare(), 100);
  109. for (unsigned i = 0; i < NElem; i++)
  110. { BOOST_CHECK(V3[i] == V2[i]);
  111. };
  112. };
  113. void test4(void)
  114. {
  115. const uint32_t KMax = 66000;
  116. std::vector<uint64_t> K, M;
  117. std::mt19937_64 my_rand(0);
  118. for (uint32_t i = 0; i < KMax; ++i)
  119. K.push_back(my_rand());
  120. M = K;
  121. bss::sample_sort(K.begin(), K.end(), 300);
  122. std::stable_sort(M.begin(), M.end());
  123. for (unsigned i = 0; i < KMax; i++)
  124. BOOST_CHECK(M[i] == K[i]);
  125. };
  126. void test5 (void)
  127. {
  128. typedef typename std::vector<xk>::iterator iter_t;
  129. std::mt19937 my_rand (0);
  130. std::vector<xk> V ;
  131. const uint32_t NELEM = 100000;
  132. V.reserve(NELEM * 10);
  133. for (uint32_t k =0 ; k < 10 ; ++k)
  134. { for ( uint32_t i =0 ; i < NELEM ; ++i)
  135. { V.emplace_back(i , k);
  136. };
  137. iter_t first = V.begin() + (k * NELEM);
  138. iter_t last = first + NELEM ;
  139. std::shuffle( first, last, my_rand);
  140. };
  141. bss::sample_sort( V.begin() , V.end());
  142. for ( uint32_t i =0 ; i < ( NELEM * 10); ++i)
  143. { BOOST_CHECK ( V[i].num == (i / 10) and V[i].tail == (i %10) );
  144. };
  145. }
  146. int test_main(int, char *[])
  147. {
  148. test1();
  149. test2();
  150. test3();
  151. test4();
  152. test5();
  153. return 0;
  154. };