171 lines
6.7 KiB
Markdown
Raw Normal View History

2021-01-29 12:31:58 +01:00
[![Arduino CI](https://github.com/RobTillaart/RunningAverage/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
2021-11-24 10:03:41 +01:00
[![Arduino-lint](https://github.com/RobTillaart/RunningAverage/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/RunningAverage/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/RunningAverage/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/RunningAverage/actions/workflows/jsoncheck.yml)
2023-11-13 17:57:14 +01:00
[![GitHub issues](https://img.shields.io/github/issues/RobTillaart/RunningAverage.svg)](https://github.com/RobTillaart/RunningAverage/issues)
2021-01-29 12:31:58 +01:00
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/RunningAverage/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/RunningAverage.svg?maxAge=3600)](https://github.com/RobTillaart/RunningAverage/releases)
2023-11-13 17:57:14 +01:00
[![PlatformIO Registry](https://badges.registry.platformio.org/packages/robtillaart/library/RunningAverage.svg)](https://registry.platformio.org/libraries/robtillaart/RunningAverage)
2021-01-29 12:31:58 +01:00
2021-11-24 10:03:41 +01:00
2020-11-27 11:33:55 +01:00
# RunningAverage
Arduino library to calculate the running average by means of a circular buffer.
2021-01-29 12:31:58 +01:00
2020-11-27 11:33:55 +01:00
## Description
2021-11-24 10:03:41 +01:00
The RunningAverage object gives a running average of the last N floating point numbers,
giving them all equal weight.
This is done by adding new data to an internal circular buffer, removing the oldest and
replace it by the newest.
The size of the internal buffer can be set in the constructor.
By keeping track of the **\_sum** the runningAverage can be calculated fast (only 1 division)
at any time. This is done with **getFastAverage()**.
2023-11-13 17:57:14 +01:00
However the constant adding and subtracting when adding new elements to the RA object possibly
introduces an ever increasing error.
2020-11-27 11:33:55 +01:00
In tests adding up to 1500000 numbers this error was always small. But that is no proof.
2021-11-24 10:03:41 +01:00
In version 0.2.16 a fix was added that uses the calculation of the sum in **getAverage()** to
2021-01-29 12:31:58 +01:00
update the internal **\_sum**.
2023-11-13 17:57:14 +01:00
#### Related
- https://github.com/RobTillaart/Correlation
- https://github.com/RobTillaart/GST - Golden standard test metrics
- https://github.com/RobTillaart/Histogram
- https://github.com/RobTillaart/RunningAngle
- https://github.com/RobTillaart/RunningAverage
- https://github.com/RobTillaart/RunningMedian
- https://github.com/RobTillaart/statHelpers - combinations & permutations
- https://github.com/RobTillaart/Statistic
2021-01-29 12:31:58 +01:00
## Interface
2023-11-13 17:57:14 +01:00
```cpp
#include "RunningAverage.h"
```
2021-01-29 12:31:58 +01:00
### Constructor
2021-11-24 10:03:41 +01:00
- **RunningAverage(uint16_t size)** allocates dynamic memory, one float (4 bytes) per element.
2021-01-29 12:31:58 +01:00
No default size (yet).
2021-11-24 10:03:41 +01:00
- **~RunningAverage()** destructor to free the memory allocated.
2021-01-29 12:31:58 +01:00
### Basic
2021-11-24 10:03:41 +01:00
- **void clear()** empties the internal buffer.
- **void add(float value)** wrapper for **addValue()**
2023-11-13 17:57:14 +01:00
- **void addValue(float value)** adds a new value to the object, if the internal buffer is full,
the oldest element is removed.
- **void fillValue(float value, uint16_t number)** adds number elements of value.
Good for initializing the system to a certain starting average.
2021-11-24 10:03:41 +01:00
- **float getValue(uint16_t position)** returns the value at **position** from the additions.
Position 0 is the first one to disappear.
- **float getAverage()** iterates over all elements to get the average, slower but accurate.
Updates the variables used by **getFastAverage()** to improve its accuracy again.
- **float getFastAverage()** reuses previous calculated values, therefore faster. Accuracy can drift.
2021-01-29 12:31:58 +01:00
### Extended functions
2021-11-24 10:03:41 +01:00
- **float getStandardDeviation()** returns the standard deviation of the current content.
Needs more than one element to be calculable.
- **float getStandardError()** returns the standard error of the current content.
- **float getMin()** returns minimum since last clear, does not need to be in the buffer any more.
- **float getMax()** returns maximum since last clear, does not need to be in the buffer any more.
- **float getMinInBuffer()** returns minimum in the internal buffer.
- **float getMaxInBuffer()** returns maximum in the internal buffer.
2021-01-29 12:31:58 +01:00
### Admin functions
2021-11-24 10:03:41 +01:00
- **bool bufferIsFull()** returns true if buffer is full.
- **float getElement(uint16_t index)** get element directly from internal buffer at index. (debug)
- **uint16_t getSize()** returns the size of the internal array.
- **uint16_t getCount()** returns the number of slots used of the internal array.
2023-11-13 17:57:14 +01:00
## Partial functions
2021-11-24 10:03:41 +01:00
- **void setPartial(uint16_t partial = 0)** use only a part of the internal array.
Allows to change the weight and history factor.
0 ==> use all == default.
- **uint16_t getPartial()** returns the set value for partial.
2023-11-13 17:57:14 +01:00
## Last functions
2021-11-24 10:03:41 +01:00
These functions get the basic statistics of the last N added elements.
Returns NAN if there are no elements and it will reduce count if there are less than
count elements in the buffer.
- **float getAverageLast(uint16_t count)** get the average of the last count elements.
- **float getMinInBufferLast(uint16_t count)** get the minimum of the last count elements.
- **float getMaxInBufferLast(uint16_t count)** get the maximum of the last count elements.
These functions are useful in cases where you might want to calculate and display the
statistics of a subset of the added elements. Reason might be to compare this with the
numbers of the whole buffer to notice changes earlier.
Otherwise one should create multiple RunningAverage objects each with its own length,
effectively having multiple copies of the data added.
2023-11-13 17:57:14 +01:00
Note: if called with a value larger or equal to **getCount()** (including **getSize()**) as
2021-11-24 10:03:41 +01:00
parameter, the functions will return the statistics of the whole buffer.
2022-11-23 19:22:02 +01:00
## Subset (experimental)
- **float getAverageSubset(uint16_t start, uint16_t count)**
Get the average of subset - count elements from start.
2021-01-29 12:31:58 +01:00
2020-11-27 11:33:55 +01:00
## Operation
See examples
2021-11-24 10:03:41 +01:00
## Future
2023-11-13 17:57:14 +01:00
#### Must
2022-11-23 19:22:02 +01:00
- update documentation, explain better
2023-11-13 17:57:14 +01:00
#### Should
2022-11-23 19:22:02 +01:00
- check for optimizations.
2023-11-13 17:57:14 +01:00
- divide by count happens often ...
2022-11-23 19:22:02 +01:00
- clear(bool zero = true) to suppress setting all to 0. ?
2023-11-13 17:57:14 +01:00
#### Could
2021-11-24 10:03:41 +01:00
- create a double based derived class? Template class?
2023-11-13 17:57:14 +01:00
- add error handling (important?).
- investigate **modus()** most frequently occurring value.
- difficult with floats ?
- what to do when on two or more values are on par?
#### Wont
- default size for constructor
- unknown what would be a good choice.
- clear(bool zero = true) to suppress setting all to 0. ?
- makes **addValue()** slightly more complex
- could introduce conflicts due to randomness data?
## Support
If you appreciate my libraries, you can support the development and maintenance.
Improve the quality of the libraries by providing issues and Pull Requests, or
donate through PayPal or GitHub sponsors.
Thank you,
2021-11-24 10:03:41 +01:00