qtconcurrentmedian.h 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. /****************************************************************************
  2. **
  3. ** Copyright (C) 2016 The Qt Company Ltd.
  4. ** Contact: https://www.qt.io/licensing/
  5. **
  6. ** This file is part of the QtCore module of the Qt Toolkit.
  7. **
  8. ** $QT_BEGIN_LICENSE:LGPL$
  9. ** Commercial License Usage
  10. ** Licensees holding valid commercial Qt licenses may use this file in
  11. ** accordance with the commercial license agreement provided with the
  12. ** Software or, alternatively, in accordance with the terms contained in
  13. ** a written agreement between you and The Qt Company. For licensing terms
  14. ** and conditions see https://www.qt.io/terms-conditions. For further
  15. ** information use the contact form at https://www.qt.io/contact-us.
  16. **
  17. ** GNU Lesser General Public License Usage
  18. ** Alternatively, this file may be used under the terms of the GNU Lesser
  19. ** General Public License version 3 as published by the Free Software
  20. ** Foundation and appearing in the file LICENSE.LGPL3 included in the
  21. ** packaging of this file. Please review the following information to
  22. ** ensure the GNU Lesser General Public License version 3 requirements
  23. ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
  24. **
  25. ** GNU General Public License Usage
  26. ** Alternatively, this file may be used under the terms of the GNU
  27. ** General Public License version 2.0 or (at your option) the GNU General
  28. ** Public license version 3 or any later version approved by the KDE Free
  29. ** Qt Foundation. The licenses are as published by the Free Software
  30. ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
  31. ** included in the packaging of this file. Please review the following
  32. ** information to ensure the GNU General Public License requirements will
  33. ** be met: https://www.gnu.org/licenses/gpl-2.0.html and
  34. ** https://www.gnu.org/licenses/gpl-3.0.html.
  35. **
  36. ** $QT_END_LICENSE$
  37. **
  38. ****************************************************************************/
  39. #ifndef QTCONCURRENT_MEDIAN_H
  40. #define QTCONCURRENT_MEDIAN_H
  41. #include <QtConcurrent/qtconcurrent_global.h>
  42. #ifndef QT_NO_CONCURRENT
  43. #include <QtCore/qvector.h>
  44. #include <algorithm>
  45. QT_BEGIN_NAMESPACE
  46. #ifndef Q_QDOC
  47. namespace QtConcurrent {
  48. template <typename T>
  49. class Median
  50. {
  51. public:
  52. Median(int _bufferSize)
  53. : currentMedian(), bufferSize(_bufferSize), currentIndex(0), valid(false), dirty(true)
  54. {
  55. values.resize(bufferSize);
  56. }
  57. void reset()
  58. {
  59. values.fill(0);
  60. currentIndex = 0;
  61. valid = false;
  62. dirty = true;
  63. }
  64. void addValue(T value)
  65. {
  66. currentIndex = ((currentIndex + 1) % bufferSize);
  67. if (valid == false && currentIndex % bufferSize == 0)
  68. valid = true;
  69. // Only update the cached median value when we have to, that
  70. // is when the new value is on then other side of the median
  71. // compared to the current value at the index.
  72. const T currentIndexValue = values[currentIndex];
  73. if ((currentIndexValue > currentMedian && currentMedian > value)
  74. || (currentMedian > currentIndexValue && value > currentMedian)) {
  75. dirty = true;
  76. }
  77. values[currentIndex] = value;
  78. }
  79. bool isMedianValid() const
  80. {
  81. return valid;
  82. }
  83. T median()
  84. {
  85. if (dirty) {
  86. dirty = false;
  87. // This is a workaround for http://gcc.gnu.org/bugzilla/show_bug.cgi?id=58800
  88. // Avoid using std::nth_element for the affected stdlibc++ releases 4.7.3 and 4.8.2.
  89. // Note that the official __GLIBCXX__ value of the releases is not used since that
  90. // one might be patched on some GNU/Linux distributions.
  91. #if defined(__GLIBCXX__) && __GLIBCXX__ <= 20140107
  92. QVector<T> sorted = values;
  93. std::sort(sorted.begin(), sorted.end());
  94. currentMedian = sorted.at(bufferSize / 2);
  95. #else
  96. QVector<T> copy = values;
  97. typename QVector<T>::iterator begin = copy.begin(), mid = copy.begin() + bufferSize/2, end = copy.end();
  98. std::nth_element(begin, mid, end);
  99. currentMedian = *mid;
  100. #endif
  101. }
  102. return currentMedian;
  103. }
  104. private:
  105. QVector<T> values;
  106. T currentMedian;
  107. int bufferSize;
  108. int currentIndex;
  109. bool valid;
  110. bool dirty;
  111. };
  112. // ### Qt6: Drop Median<double> in favor of this faster MedianDouble
  113. class MedianDouble
  114. {
  115. public:
  116. enum { BufferSize = 7 };
  117. MedianDouble()
  118. : currentMedian(), currentIndex(0), valid(false), dirty(true)
  119. {
  120. }
  121. void reset()
  122. {
  123. std::fill_n(values, static_cast<int>(BufferSize), 0.0);
  124. currentIndex = 0;
  125. valid = false;
  126. dirty = true;
  127. }
  128. void addValue(double value)
  129. {
  130. ++currentIndex;
  131. if (currentIndex == BufferSize) {
  132. currentIndex = 0;
  133. valid = true;
  134. }
  135. // Only update the cached median value when we have to, that
  136. // is when the new value is on then other side of the median
  137. // compared to the current value at the index.
  138. const double currentIndexValue = values[currentIndex];
  139. if ((currentIndexValue > currentMedian && currentMedian > value)
  140. || (currentMedian > currentIndexValue && value > currentMedian)) {
  141. dirty = true;
  142. }
  143. values[currentIndex] = value;
  144. }
  145. bool isMedianValid() const
  146. {
  147. return valid;
  148. }
  149. double median()
  150. {
  151. if (dirty) {
  152. dirty = false;
  153. double sorted[BufferSize];
  154. ::memcpy(&sorted, &values, sizeof(sorted));
  155. std::sort(sorted, sorted + static_cast<int>(BufferSize));
  156. currentMedian = sorted[BufferSize / 2];
  157. }
  158. return currentMedian;
  159. }
  160. private:
  161. double values[BufferSize];
  162. double currentMedian;
  163. int currentIndex;
  164. bool valid;
  165. bool dirty;
  166. };
  167. } // namespace QtConcurrent
  168. #endif //Q_QDOC
  169. QT_END_NAMESPACE
  170. #endif // QT_NO_CONCURRENT
  171. #endif