123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204 |
- /****************************************************************************
- **
- ** Copyright (C) 2016 The Qt Company Ltd.
- ** Contact: https://www.qt.io/licensing/
- **
- ** This file is part of the QtCore module of the Qt Toolkit.
- **
- ** $QT_BEGIN_LICENSE:LGPL$
- ** Commercial License Usage
- ** Licensees holding valid commercial Qt licenses may use this file in
- ** accordance with the commercial license agreement provided with the
- ** Software or, alternatively, in accordance with the terms contained in
- ** a written agreement between you and The Qt Company. For licensing terms
- ** and conditions see https://www.qt.io/terms-conditions. For further
- ** information use the contact form at https://www.qt.io/contact-us.
- **
- ** GNU Lesser General Public License Usage
- ** Alternatively, this file may be used under the terms of the GNU Lesser
- ** General Public License version 3 as published by the Free Software
- ** Foundation and appearing in the file LICENSE.LGPL3 included in the
- ** packaging of this file. Please review the following information to
- ** ensure the GNU Lesser General Public License version 3 requirements
- ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
- **
- ** GNU General Public License Usage
- ** Alternatively, this file may be used under the terms of the GNU
- ** General Public License version 2.0 or (at your option) the GNU General
- ** Public license version 3 or any later version approved by the KDE Free
- ** Qt Foundation. The licenses are as published by the Free Software
- ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
- ** included in the packaging of this file. Please review the following
- ** information to ensure the GNU General Public License requirements will
- ** be met: https://www.gnu.org/licenses/gpl-2.0.html and
- ** https://www.gnu.org/licenses/gpl-3.0.html.
- **
- ** $QT_END_LICENSE$
- **
- ****************************************************************************/
- #ifndef QTCONCURRENT_MEDIAN_H
- #define QTCONCURRENT_MEDIAN_H
- #include <QtConcurrent/qtconcurrent_global.h>
- #ifndef QT_NO_CONCURRENT
- #include <QtCore/qvector.h>
- #include <algorithm>
- QT_BEGIN_NAMESPACE
- #ifndef Q_QDOC
- namespace QtConcurrent {
- template <typename T>
- class Median
- {
- public:
- Median(int _bufferSize)
- : currentMedian(), bufferSize(_bufferSize), currentIndex(0), valid(false), dirty(true)
- {
- values.resize(bufferSize);
- }
- void reset()
- {
- values.fill(0);
- currentIndex = 0;
- valid = false;
- dirty = true;
- }
- void addValue(T value)
- {
- currentIndex = ((currentIndex + 1) % bufferSize);
- if (valid == false && currentIndex % bufferSize == 0)
- valid = true;
- // Only update the cached median value when we have to, that
- // is when the new value is on then other side of the median
- // compared to the current value at the index.
- const T currentIndexValue = values[currentIndex];
- if ((currentIndexValue > currentMedian && currentMedian > value)
- || (currentMedian > currentIndexValue && value > currentMedian)) {
- dirty = true;
- }
- values[currentIndex] = value;
- }
- bool isMedianValid() const
- {
- return valid;
- }
- T median()
- {
- if (dirty) {
- dirty = false;
- // This is a workaround for http://gcc.gnu.org/bugzilla/show_bug.cgi?id=58800
- // Avoid using std::nth_element for the affected stdlibc++ releases 4.7.3 and 4.8.2.
- // Note that the official __GLIBCXX__ value of the releases is not used since that
- // one might be patched on some GNU/Linux distributions.
- #if defined(__GLIBCXX__) && __GLIBCXX__ <= 20140107
- QVector<T> sorted = values;
- std::sort(sorted.begin(), sorted.end());
- currentMedian = sorted.at(bufferSize / 2);
- #else
- QVector<T> copy = values;
- typename QVector<T>::iterator begin = copy.begin(), mid = copy.begin() + bufferSize/2, end = copy.end();
- std::nth_element(begin, mid, end);
- currentMedian = *mid;
- #endif
- }
- return currentMedian;
- }
- private:
- QVector<T> values;
- T currentMedian;
- int bufferSize;
- int currentIndex;
- bool valid;
- bool dirty;
- };
- // ### Qt6: Drop Median<double> in favor of this faster MedianDouble
- class MedianDouble
- {
- public:
- enum { BufferSize = 7 };
- MedianDouble()
- : currentMedian(), currentIndex(0), valid(false), dirty(true)
- {
- }
- void reset()
- {
- std::fill_n(values, static_cast<int>(BufferSize), 0.0);
- currentIndex = 0;
- valid = false;
- dirty = true;
- }
- void addValue(double value)
- {
- ++currentIndex;
- if (currentIndex == BufferSize) {
- currentIndex = 0;
- valid = true;
- }
- // Only update the cached median value when we have to, that
- // is when the new value is on then other side of the median
- // compared to the current value at the index.
- const double currentIndexValue = values[currentIndex];
- if ((currentIndexValue > currentMedian && currentMedian > value)
- || (currentMedian > currentIndexValue && value > currentMedian)) {
- dirty = true;
- }
- values[currentIndex] = value;
- }
- bool isMedianValid() const
- {
- return valid;
- }
- double median()
- {
- if (dirty) {
- dirty = false;
- double sorted[BufferSize];
- ::memcpy(&sorted, &values, sizeof(sorted));
- std::sort(sorted, sorted + static_cast<int>(BufferSize));
- currentMedian = sorted[BufferSize / 2];
- }
- return currentMedian;
- }
- private:
- double values[BufferSize];
- double currentMedian;
- int currentIndex;
- bool valid;
- bool dirty;
- };
- } // namespace QtConcurrent
- #endif //Q_QDOC
- QT_END_NAMESPACE
- #endif // QT_NO_CONCURRENT
- #endif
|