2011-10-09 22:24:29 +02:00
|
|
|
//
|
|
|
|
// FILE: RunningMedian.cpp
|
2021-01-29 12:31:58 +01:00
|
|
|
// AUTHOR: Rob Tillaart
|
|
|
|
// VERSION: 0.3.3
|
2011-10-09 22:24:29 +02:00
|
|
|
// PURPOSE: RunningMedian library for Arduino
|
|
|
|
//
|
2021-01-29 12:31:58 +01:00
|
|
|
// HISTORY:
|
|
|
|
// 0.1.00 2011-02-16 initial version
|
|
|
|
// 0.1.01 2011-02-22 added remarks from CodingBadly
|
|
|
|
// 0.1.02 2012-03-15 added
|
|
|
|
// 0.1.03 2013-09-30 added _sorted flag, minor refactor
|
|
|
|
// 0.1.04 2013-10-17 added getAverage(uint8_t) - kudo's to Sembazuru
|
|
|
|
// 0.1.05 2013-10-18 fixed bug in sort; removes default constructor; dynamic memory
|
|
|
|
// 0.1.06 2013-10-19 faster sort, dynamic arrays, replaced sorted float array with indirection array
|
|
|
|
// 0.1.07 2013-10-19 add correct median if _count is even.
|
|
|
|
// 0.1.08 2013-10-20 add getElement(), add getSottedElement() add predict()
|
|
|
|
// 0.1.09 2014-11-25 float to double (support ARM)
|
|
|
|
// 0.1.10 2015-03-07 fix clear
|
|
|
|
// 0.1.11 2015-03-29 undo 0.1.10 fix clear
|
|
|
|
// 0.1.12 2015-07-12 refactor constructor + const
|
|
|
|
// 0.1.13 2015-10-30 fix getElement(n) - kudos to Gdunge
|
|
|
|
// 0.1.14 2017-07-26 revert double to float - issue #33
|
|
|
|
// 0.1.15 2018-08-24 make runningMedian Configurable #110
|
|
|
|
// 0.2.0 2020-04-16 refactor.
|
|
|
|
// 0.2.1 2020-06-19 fix library.json
|
|
|
|
// 0.2.2 2021-01-03 add Arduino-CI + unit tests
|
|
|
|
// 0.3.0 2021-01-04 malloc memory as default storage
|
|
|
|
// 0.3.1 2021-01-16 Changed size parameter to 255 max
|
|
|
|
// 0.3.2 2021-01-21 replaced bubbleSort by insertionSort
|
|
|
|
// --> better performance for large arrays.
|
|
|
|
// 0.3.3 2021-01-22 better insertionSort (+ cleanup test code)
|
|
|
|
|
2011-10-09 22:24:29 +02:00
|
|
|
|
|
|
|
#include "RunningMedian.h"
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2015-10-30 16:46:22 +01:00
|
|
|
RunningMedian::RunningMedian(const uint8_t size)
|
2011-10-09 22:24:29 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
_size = size;
|
|
|
|
if (_size < MEDIAN_MIN_SIZE) _size = MEDIAN_MIN_SIZE;
|
|
|
|
// if (_size > MEDIAN_MAX_SIZE) _size = MEDIAN_MAX_SIZE;
|
2013-10-19 15:55:43 +02:00
|
|
|
|
2013-10-19 00:26:50 +02:00
|
|
|
#ifdef RUNNING_MEDIAN_USE_MALLOC
|
2021-01-29 12:31:58 +01:00
|
|
|
_values = (float *) malloc(_size * sizeof(float));
|
|
|
|
_sortIdx = (uint8_t *) malloc(_size * sizeof(uint8_t));
|
2013-10-19 00:26:50 +02:00
|
|
|
#endif
|
2017-07-26 22:21:25 +02:00
|
|
|
clear();
|
2011-10-09 22:24:29 +02:00
|
|
|
}
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2013-10-19 00:26:50 +02:00
|
|
|
RunningMedian::~RunningMedian()
|
2013-08-17 14:54:10 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
#ifdef RUNNING_MEDIAN_USE_MALLOC
|
|
|
|
free(_values);
|
|
|
|
free(_sortIdx);
|
|
|
|
#endif
|
2013-08-17 14:54:10 +02:00
|
|
|
}
|
2013-10-19 00:26:50 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
// resets all internal counters
|
2011-10-09 22:24:29 +02:00
|
|
|
void RunningMedian::clear()
|
2013-10-17 20:55:03 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
_count = 0;
|
|
|
|
_index = 0;
|
2017-07-26 22:21:25 +02:00
|
|
|
_sorted = false;
|
2020-11-27 11:33:55 +01:00
|
|
|
for (uint8_t i = 0; i < _size; i++)
|
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
_sortIdx[i] = i;
|
2020-11-27 11:33:55 +01:00
|
|
|
}
|
2011-10-09 22:24:29 +02:00
|
|
|
}
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2011-10-09 22:24:29 +02:00
|
|
|
// adds a new value to the data-set
|
|
|
|
// or overwrites the oldest if full.
|
2017-07-26 22:21:25 +02:00
|
|
|
void RunningMedian::add(float value)
|
2011-10-09 22:24:29 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
_values[_index++] = value;
|
|
|
|
if (_index >= _size) _index = 0; // wrap around
|
|
|
|
if (_count < _size) _count++;
|
2017-07-26 22:21:25 +02:00
|
|
|
_sorted = false;
|
2011-10-09 22:24:29 +02:00
|
|
|
}
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2017-07-26 22:21:25 +02:00
|
|
|
float RunningMedian::getMedian()
|
2011-10-09 22:24:29 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
if (_count == 0) return NAN;
|
2017-07-26 22:21:25 +02:00
|
|
|
|
|
|
|
if (_sorted == false) sort();
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
if (_count & 0x01) // is it odd sized?
|
2020-11-27 11:33:55 +01:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
return _values[_sortIdx[_count / 2]];
|
2020-11-27 11:33:55 +01:00
|
|
|
}
|
2021-01-29 12:31:58 +01:00
|
|
|
return (_values[_sortIdx[_count / 2]] + _values[_sortIdx[_count / 2 - 1]]) / 2;
|
2020-11-27 11:33:55 +01:00
|
|
|
}
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2020-11-27 11:33:55 +01:00
|
|
|
float RunningMedian::getQuantile(float q)
|
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
if (_count == 0) return NAN;
|
2020-11-27 11:33:55 +01:00
|
|
|
|
|
|
|
if ((q < 0) || (q > 1)) return NAN;
|
|
|
|
|
|
|
|
if (_sorted == false) sort();
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
const float id = (_count - 1) * q;
|
|
|
|
const uint8_t lo = floor(id);
|
|
|
|
const uint8_t hi = ceil(id);
|
|
|
|
const float qs = _values[_sortIdx[lo]];
|
|
|
|
const float h = (id - lo);
|
2020-11-27 11:33:55 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
return (1.0 - h) * qs + h * _values[_sortIdx[hi]];
|
2011-10-09 22:24:29 +02:00
|
|
|
}
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2017-07-26 22:21:25 +02:00
|
|
|
float RunningMedian::getAverage()
|
2015-10-30 16:46:22 +01:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
if (_count == 0) return NAN;
|
2013-08-17 14:54:10 +02:00
|
|
|
|
2017-07-26 22:21:25 +02:00
|
|
|
float sum = 0;
|
2021-01-29 12:31:58 +01:00
|
|
|
for (uint8_t i = 0; i < _count; i++)
|
2020-11-27 11:33:55 +01:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
sum += _values[i];
|
2020-11-27 11:33:55 +01:00
|
|
|
}
|
2021-01-29 12:31:58 +01:00
|
|
|
return sum / _count;
|
2015-10-30 16:46:22 +01:00
|
|
|
}
|
2013-08-17 14:54:10 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2017-07-26 22:21:25 +02:00
|
|
|
float RunningMedian::getAverage(uint8_t nMedians)
|
2013-08-17 14:54:10 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
if ((_count == 0) || (nMedians == 0)) return NAN;
|
2013-10-17 20:55:03 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
if (_count < nMedians) nMedians = _count; // when filling the array for first time
|
|
|
|
uint8_t start = ((_count - nMedians) / 2);
|
2017-07-26 22:21:25 +02:00
|
|
|
uint8_t stop = start + nMedians;
|
|
|
|
|
|
|
|
if (_sorted == false) sort();
|
|
|
|
|
|
|
|
float sum = 0;
|
2020-11-27 11:33:55 +01:00
|
|
|
for (uint8_t i = start; i < stop; i++)
|
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
sum += _values[_sortIdx[i]];
|
2020-11-27 11:33:55 +01:00
|
|
|
}
|
2017-07-26 22:21:25 +02:00
|
|
|
return sum / nMedians;
|
2013-08-17 14:54:10 +02:00
|
|
|
}
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2017-07-26 22:21:25 +02:00
|
|
|
float RunningMedian::getElement(const uint8_t n)
|
2013-10-20 15:32:13 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
if ((_count == 0) || (n >= _count)) return NAN;
|
2017-07-26 22:21:25 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
uint8_t pos = _index + n;
|
|
|
|
if (pos >= _count) // faster than %
|
2017-07-26 22:21:25 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
pos -= _count;
|
2017-07-26 22:21:25 +02:00
|
|
|
}
|
2021-01-29 12:31:58 +01:00
|
|
|
return _values[pos];
|
2013-10-20 15:32:13 +02:00
|
|
|
}
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2017-07-26 22:21:25 +02:00
|
|
|
float RunningMedian::getSortedElement(const uint8_t n)
|
2013-10-20 15:32:13 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
if ((_count == 0) || (n >= _count)) return NAN;
|
2017-07-26 22:21:25 +02:00
|
|
|
|
|
|
|
if (_sorted == false) sort();
|
2021-01-29 12:31:58 +01:00
|
|
|
return _values[_sortIdx[n]];
|
2013-10-20 15:32:13 +02:00
|
|
|
}
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2015-03-07 15:53:38 +01:00
|
|
|
// n can be max <= half the (filled) size
|
2017-07-26 22:21:25 +02:00
|
|
|
float RunningMedian::predict(const uint8_t n)
|
2013-10-20 15:32:13 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
uint8_t mid = _count / 2;
|
|
|
|
if ((_count == 0) || (n >= mid)) return NAN;
|
2017-07-26 22:21:25 +02:00
|
|
|
|
|
|
|
float med = getMedian(); // takes care of sorting !
|
2021-01-29 12:31:58 +01:00
|
|
|
if (_count & 0x01) // odd # elements
|
2017-07-26 22:21:25 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
return max(med - _values[_sortIdx[mid - n]], _values[_sortIdx[mid + n]] - med);
|
2017-07-26 22:21:25 +02:00
|
|
|
}
|
2021-01-29 12:31:58 +01:00
|
|
|
// even # elements
|
|
|
|
float f1 = (_values[_sortIdx[mid - n]] + _values[_sortIdx[mid - n - 1]]) / 2;
|
|
|
|
float f2 = (_values[_sortIdx[mid + n]] + _values[_sortIdx[mid + n - 1]]) / 2;
|
2020-11-27 11:33:55 +01:00
|
|
|
return max(med - f1, f2 - med) / 2;
|
2013-10-20 15:32:13 +02:00
|
|
|
}
|
2013-10-17 20:55:03 +02:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2011-10-09 22:24:29 +02:00
|
|
|
void RunningMedian::sort()
|
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
// insertSort
|
|
|
|
for (uint16_t i = 1; i < _count; i++)
|
2017-07-26 22:21:25 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
uint16_t z = i;
|
|
|
|
uint16_t temp = _sortIdx[z];
|
|
|
|
while ((z > 0) && (_values[temp] < _values[_sortIdx[z - 1]]))
|
2013-10-17 20:55:03 +02:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
_sortIdx[z] = _sortIdx[z - 1];
|
|
|
|
z--;
|
2013-10-17 20:55:03 +02:00
|
|
|
}
|
2021-01-29 12:31:58 +01:00
|
|
|
_sortIdx[z] = temp;
|
2017-07-26 22:21:25 +02:00
|
|
|
}
|
|
|
|
_sorted = true;
|
2011-10-09 22:24:29 +02:00
|
|
|
}
|
|
|
|
|
2020-11-27 11:33:55 +01:00
|
|
|
// -- END OF FILE --
|