Main.cpp 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. #include <chrono>
  2. #include <iostream>
  3. long getNanos()
  4. {
  5. return std::chrono::high_resolution_clock::now().time_since_epoch().count();
  6. }
  7. int intKey(const int &i) { return i; }
  8. static unsigned long seed = 0;
  9. int randomInt()
  10. {
  11. seed = (seed * 0x5DEECE66DL + 0xBL) & 0xFFFFFFFFFFFF;
  12. return static_cast<int>(seed >> 17);
  13. }
  14. template <typename T> T *mergeSort(const T *in, int size, int (*key)(const T &))
  15. {
  16. if (size <= 1)
  17. {
  18. T *out = new T[size];
  19. for (int i = 0; i < size; ++i)
  20. {
  21. out[i] = in[i];
  22. }
  23. return out;
  24. }
  25. // split
  26. int sizeLeft = size / 2;
  27. int sizeRight = (size + 1) / 2;
  28. const T *left = in;
  29. const T *right = in + sizeLeft;
  30. left = mergeSort(left, sizeLeft, key);
  31. right = mergeSort(right, sizeRight, key);
  32. T *out = new T[size];
  33. // merge
  34. int rightCounter = 0;
  35. int leftCounter = 0;
  36. for (int i = 0; i < size; ++i)
  37. {
  38. if (leftCounter >= sizeLeft)
  39. {
  40. out[i] = right[rightCounter];
  41. ++rightCounter;
  42. }
  43. else if (rightCounter >= sizeRight)
  44. {
  45. out[i] = left[leftCounter];
  46. ++leftCounter;
  47. }
  48. else if (key(left[leftCounter]) < key(right[rightCounter]))
  49. {
  50. out[i] = left[leftCounter];
  51. ++leftCounter;
  52. }
  53. else
  54. {
  55. out[i] = right[rightCounter];
  56. ++rightCounter;
  57. }
  58. }
  59. delete[] right;
  60. delete[] left;
  61. return out;
  62. }
  63. void cacheTest()
  64. {
  65. for (int size = 512; size < 100'000'000; size *= 2)
  66. {
  67. printf("Cache Test with size = %9d | ", size);
  68. constexpr int max = 500;
  69. int *a = new int[size];
  70. seed = 0;
  71. for (int i = 0; i < size; i++)
  72. {
  73. a[i] = randomInt() % max;
  74. }
  75. long time = -getNanos();
  76. int *b = mergeSort(a, size, intKey);
  77. time += getNanos();
  78. printf("%10ld ns | %7.2lf ns / entry \n", time,
  79. (static_cast<double>(time) / size));
  80. delete[] a;
  81. delete[] b;
  82. }
  83. }
  84. struct A
  85. {
  86. int key;
  87. char filler[32];
  88. };
  89. int aKey(const A &a) { return a.key; }
  90. void structTest()
  91. {
  92. for (int size = 512; size < 100'000'000; size *= 2)
  93. {
  94. printf("Struct Test with size = %9d | ", size);
  95. constexpr int max = 500;
  96. A *a = new A[size];
  97. seed = 0;
  98. for (int i = 0; i < size; i++)
  99. {
  100. a[i].key = randomInt() % max;
  101. }
  102. long time = -getNanos();
  103. A *b = mergeSort(a, size, aKey);
  104. time += getNanos();
  105. printf("%10ld ns | %7.2lf ns / entry \n", time,
  106. (static_cast<double>(time) / size));
  107. delete[] a;
  108. delete[] b;
  109. }
  110. }
  111. int pointerKey(int *const &a) { return *a; }
  112. void pointerTest()
  113. {
  114. for (int size = 512; size < 100'000'000; size *= 2)
  115. {
  116. printf("Pointer Test with size = %9d | ", size);
  117. constexpr int max = 500;
  118. int **a = new int *[size];
  119. seed = 0;
  120. for (int i = 0; i < size; i++)
  121. {
  122. a[i] = new int(randomInt() % max);
  123. }
  124. long time = -getNanos();
  125. int **b = mergeSort<int *>(a, size, pointerKey);
  126. time += getNanos();
  127. printf("%10ld ns | %7.2lf ns / entry \n", time,
  128. (static_cast<double>(time) / size));
  129. for (int i = 0; i < size; i++)
  130. {
  131. delete a[i];
  132. }
  133. delete[] a;
  134. delete[] b;
  135. }
  136. }
  137. int main()
  138. {
  139. cacheTest();
  140. structTest();
  141. pointerTest();
  142. return 0;
  143. }