2011-10-09 22:24:29 +02:00
|
|
|
//
|
|
|
|
// FILE: RunningMedian.cpp
|
2021-01-29 12:31:58 +01:00
|
|
|
// AUTHOR: Rob Tillaart
|
2023-07-13 14:27:45 +02:00
|
|
|
// VERSION: 0.3.8
|
2011-10-09 22:24:29 +02:00
|
|
|
// PURPOSE: RunningMedian library for Arduino
|
|
|
|
//
|
2022-11-06 10:24:44 +01:00
|
|
|
// HISTORY: see changelog.md
|
2021-01-29 12:31:58 +01:00
|
|
|
|
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;
|
2022-06-06 10:32:23 +02:00
|
|
|
#if !RUNNING_MEDIAN_USE_MALLOC
|
|
|
|
if (_size > MEDIAN_MAX_SIZE) _size = MEDIAN_MAX_SIZE;
|
|
|
|
#endif
|
2013-10-19 15:55:43 +02:00
|
|
|
|
2022-06-06 10:32:23 +02:00
|
|
|
#if 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
|
|
|
{
|
2022-06-06 10:32:23 +02:00
|
|
|
#if RUNNING_MEDIAN_USE_MALLOC
|
2021-01-29 12:31:58 +01:00
|
|
|
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
|
|
|
|
2022-11-06 10:24:44 +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
|
|
|
|
2022-11-06 10:24:44 +01: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;
|
2022-11-06 10:24:44 +01:00
|
|
|
if (_index >= _size) _index = 0; // wrap around
|
2021-01-29 12:31:58 +01:00
|
|
|
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();
|
|
|
|
|
2022-11-06 10:24:44 +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
|
|
|
|
2021-12-28 10:25:17 +01:00
|
|
|
float RunningMedian::getQuantile(float quantile)
|
2020-11-27 11:33:55 +01:00
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
if (_count == 0) return NAN;
|
2021-12-28 10:25:17 +01:00
|
|
|
|
|
|
|
if ((quantile < 0) || (quantile > 1)) return NAN;
|
2020-11-27 11:33:55 +01:00
|
|
|
|
|
|
|
if (_sorted == false) sort();
|
2021-12-28 10:25:17 +01:00
|
|
|
|
|
|
|
const float index = (_count - 1) * quantile;
|
|
|
|
const uint8_t lo = floor(index);
|
|
|
|
const uint8_t hi = ceil(index);
|
|
|
|
const float qs = _values[_sortIdx[lo]];
|
|
|
|
const float h = (index - 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
|
|
|
|
2022-11-06 10:24:44 +01:00
|
|
|
// when filling the array for first time
|
|
|
|
if (_count < nMedians) nMedians = _count;
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
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
|
|
|
|
2023-07-13 14:27:45 +02:00
|
|
|
// nMedians is the spread, or the middle N
|
|
|
|
// this version compensated for bias #22
|
|
|
|
|
|
|
|
float RunningMedian::getMedianAverage(uint8_t nMedians)
|
|
|
|
{
|
|
|
|
// handle special cases.
|
|
|
|
if ((_count == 0) || (nMedians == 0)) return NAN;
|
|
|
|
if (_count == 1) return _values[0];
|
|
|
|
if (_count == 2) return (_values[0] + _values[1]) * 0.5;
|
|
|
|
|
|
|
|
// nMedians can not be larger than current nr of elements.
|
|
|
|
if (_count <= nMedians) return getAverage();
|
|
|
|
|
|
|
|
// _count is at least 3 from here
|
|
|
|
|
|
|
|
// Eliminate the bias when the nMedians would fall slightly
|
|
|
|
// to the left or right of the centre.
|
|
|
|
// If count and nMedians are not both odd or both even reduce
|
|
|
|
// the spread by 1 to make them the same.
|
|
|
|
// If nMedians becomes 0 correct this. to 2.
|
|
|
|
if ((_count & 0x01) != (nMedians & 0x01))
|
|
|
|
{
|
|
|
|
--nMedians;
|
|
|
|
// nmedians can not become 0
|
|
|
|
if (nMedians == 0) nMedians = 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t start = (_count - nMedians) / 2;
|
|
|
|
uint8_t stop = start + nMedians;
|
|
|
|
|
|
|
|
if (_sorted == false) sort();
|
|
|
|
|
|
|
|
float sum = 0;
|
|
|
|
for (uint8_t i = start; i < stop; i++)
|
|
|
|
{
|
|
|
|
sum += _values[_sortIdx[i]];
|
|
|
|
}
|
|
|
|
return sum / nMedians;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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;
|
2022-11-06 10:24:44 +01:00
|
|
|
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
|
|
|
|
2022-11-06 10:24:44 +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
|
|
|
|
2022-11-06 10:24:44 +01:00
|
|
|
float med = getMedian(); // takes care of sorting !
|
|
|
|
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
|
|
|
}
|
2022-11-06 10:24:44 +01:00
|
|
|
// even # elements
|
2021-01-29 12:31:58 +01:00
|
|
|
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
|
|
|
|
2022-11-06 10:24:44 +01:00
|
|
|
void RunningMedian::setSearchMode(uint8_t searchMode)
|
|
|
|
{
|
|
|
|
if (searchMode == 1) _searchMode = 1;
|
|
|
|
else _searchMode = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t RunningMedian::getSearchMode()
|
|
|
|
{
|
|
|
|
return _searchMode;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// PRIVATE
|
|
|
|
//
|
|
|
|
|
|
|
|
// insertion sort - _searchMode = linear or binary.
|
|
|
|
|
|
|
|
void RunningMedian::sort()
|
|
|
|
{
|
|
|
|
uint16_t lo = 0;
|
|
|
|
uint16_t hi = 0;
|
|
|
|
uint16_t mi = 0;
|
|
|
|
uint16_t temp = 0;
|
|
|
|
|
|
|
|
for (uint16_t i = 1; i < _count; i++)
|
|
|
|
{
|
|
|
|
temp = _sortIdx[i];
|
|
|
|
float f = _values[temp];
|
|
|
|
|
|
|
|
// handle special case f is smaller than all elements first.
|
|
|
|
// only one compare needed, improves linear search too.
|
|
|
|
if (f <= _values[_sortIdx[0]])
|
|
|
|
{
|
|
|
|
hi = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (_searchMode == 0)
|
|
|
|
{
|
|
|
|
hi = i;
|
|
|
|
// find insertion point with linear search
|
|
|
|
while ((hi > 0) && (f < _values[_sortIdx[hi - 1]]))
|
|
|
|
{
|
|
|
|
hi--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (_searchMode == 1)
|
|
|
|
{
|
|
|
|
// find insertion point with binary search
|
|
|
|
lo = 0;
|
|
|
|
hi = i;
|
|
|
|
// be aware there might be duplicates
|
|
|
|
while (hi - lo > 1)
|
|
|
|
{
|
|
|
|
mi = (lo + hi) / 2;
|
|
|
|
if (f < _values[_sortIdx[mi]])
|
|
|
|
{
|
|
|
|
hi = mi;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
lo = mi;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// move elements to make space
|
|
|
|
uint16_t k = i;
|
|
|
|
while (k > hi)
|
|
|
|
{
|
|
|
|
_sortIdx[k] = _sortIdx[k - 1];
|
|
|
|
k--;
|
|
|
|
}
|
|
|
|
|
|
|
|
// insert at right spot.
|
|
|
|
_sortIdx[k] = temp;
|
|
|
|
|
|
|
|
}
|
|
|
|
_sorted = true;
|
|
|
|
|
|
|
|
// // verify sorted
|
|
|
|
// for (int i = 0; i < _count; i++)
|
|
|
|
// {
|
|
|
|
// if (i%5 == 0) Serial.println();
|
|
|
|
// Serial.print("\t");
|
|
|
|
// Serial.print(_values[_sortIdx[i]]);
|
|
|
|
// }
|
|
|
|
// Serial.println("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
split version of pre-0.3.7 sort - bit faster
|
|
|
|
|
2011-10-09 22:24:29 +02:00
|
|
|
void RunningMedian::sort()
|
|
|
|
{
|
2021-12-28 10:25:17 +01:00
|
|
|
// insertSort
|
2022-11-06 10:24:44 +01:00
|
|
|
for (uint16_t i = 1; i < _count; i++)
|
|
|
|
{
|
|
|
|
uint16_t hi = i;
|
|
|
|
uint16_t temp = _sortIdx[hi];
|
|
|
|
float f = _values[temp];
|
|
|
|
while ((hi > 0) && (f < _values[_sortIdx[hi - 1]]))
|
|
|
|
{
|
|
|
|
hi--;
|
|
|
|
}
|
|
|
|
|
|
|
|
// move elements to make space
|
|
|
|
uint16_t k = i;
|
|
|
|
while (k > hi)
|
|
|
|
{
|
|
|
|
_sortIdx[k] = _sortIdx[k - 1];
|
|
|
|
k--;
|
|
|
|
}
|
|
|
|
|
|
|
|
// insert at right spot.
|
|
|
|
_sortIdx[k] = temp;
|
|
|
|
}
|
|
|
|
_sorted = true;
|
|
|
|
// // verify sorted
|
|
|
|
// for (int i = 0; i < _count; i++)
|
|
|
|
// {
|
|
|
|
// if (i%5 == 0) Serial.println();
|
|
|
|
// Serial.print("\t");
|
|
|
|
// Serial.print(_values[_sortIdx[i]]);
|
|
|
|
// }
|
|
|
|
// Serial.println("\n");
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
// straightforward insertion sort - PRE-0.3.7
|
|
|
|
void RunningMedian::sort()
|
|
|
|
{
|
2021-01-29 12:31:58 +01:00
|
|
|
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
|
|
|
|
2022-11-06 10:24:44 +01:00
|
|
|
// // verify sorted
|
|
|
|
// for (int i = 0; i < _count; i++)
|
|
|
|
// {
|
|
|
|
// if (i%5 == 0) Serial.println();
|
|
|
|
// Serial.print("\t");
|
|
|
|
// Serial.print(_values[_sortIdx[i]]);
|
|
|
|
// }
|
|
|
|
// Serial.println("\n");
|
|
|
|
}
|
|
|
|
*/
|
2021-12-28 10:25:17 +01:00
|
|
|
|
2020-11-27 11:33:55 +01:00
|
|
|
// -- END OF FILE --
|
2021-12-28 10:25:17 +01:00
|
|
|
|