2023-05-18 21:21:08 +02:00
|
|
|
|
|
|
|
[![Arduino CI](https://github.com/RobTillaart/TOPMAX/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
|
|
|
|
[![Arduino-lint](https://github.com/RobTillaart/TOPMAX/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/TOPMAX/actions/workflows/arduino-lint.yml)
|
|
|
|
[![JSON check](https://github.com/RobTillaart/TOPMAX/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/TOPMAX/actions/workflows/jsoncheck.yml)
|
|
|
|
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/TOPMAX/blob/master/LICENSE)
|
|
|
|
[![GitHub release](https://img.shields.io/github/release/RobTillaart/TOPMAX.svg?maxAge=3600)](https://github.com/RobTillaart/TOPMAX/releases)
|
|
|
|
|
|
|
|
|
|
|
|
# TOPMAX
|
|
|
|
|
|
|
|
Arduino library to track the top N maxima.
|
|
|
|
|
|
|
|
## Description
|
|
|
|
|
|
|
|
This experimental library tracks the top N maxima of a series of values.
|
2023-06-13 19:16:10 +02:00
|
|
|
|
|
|
|
The library implements two classes:
|
|
|
|
- **TOPMAX** just tracks the values. (smaller footprint).
|
|
|
|
- **TOPMAXext** also tracks a user defined 32-bit tag (e.g. timestamp).
|
|
|
|
|
|
|
|
The latter is more useful as it allows to link the top N values to a certain time
|
|
|
|
or day, an index in a database, or another indirect reference.
|
|
|
|
It can even be used to store a second parameter e.g. what was the humidity for
|
|
|
|
the top N temperatures.
|
|
|
|
|
|
|
|
The library can be used for different applications:
|
|
|
|
- track top N peak temperature / humidity over a given period of time.
|
|
|
|
- track top N UV peak intensity during the day.
|
|
|
|
- track top N trading volumes or $$.
|
|
|
|
- track top N largest fish caught in a contest.
|
|
|
|
- track top N number of visitors during a day.
|
|
|
|
- track top N of time-outs, e.g. to monitor production loss.
|
|
|
|
|
|
|
|
The 32 bit tag, can be filled with a 32 bit number like a timestamp, bit can also
|
|
|
|
be used more creatively e.g. 2x 16 bit numbers or even 32 booleans.
|
|
|
|
Any mapping is possible (but not part of the library).
|
2023-05-18 21:21:08 +02:00
|
|
|
|
|
|
|
|
2023-06-17 18:01:24 +02:00
|
|
|
#### 0.2.0 breaking change
|
|
|
|
|
|
|
|
Since version 0.2.0 the maxima is found with **getValue(index == 0)**.
|
|
|
|
In earlier versions the maxima was found with **index == count()**.
|
|
|
|
|
|
|
|
The advantage of this change is that independent of the size of the TOPMAX
|
|
|
|
object the true maxima will always at index == 0. So if you change your code
|
|
|
|
and the size of the TOPMAX object, far less code needs to be changed.
|
|
|
|
It also allows to have two or more TOPMAX objects of different size and use one
|
|
|
|
index to access both.
|
|
|
|
|
|
|
|
|
2023-05-18 21:21:08 +02:00
|
|
|
#### Links
|
|
|
|
|
|
|
|
- https://github.com/RobTillaart/TOPMAX
|
|
|
|
- https://github.com/RobTillaart/TOPMIN
|
|
|
|
- https://github.com/RobTillaart/runningAverage
|
|
|
|
- https://github.com/RobTillaart/MINMAX
|
|
|
|
|
|
|
|
|
|
|
|
## Interface
|
|
|
|
|
|
|
|
```cpp
|
|
|
|
#include "TOPMAX.h"
|
|
|
|
```
|
|
|
|
|
2023-06-13 19:16:10 +02:00
|
|
|
#### TOPMAX
|
|
|
|
|
2023-05-18 21:21:08 +02:00
|
|
|
- **TOPMAX(uint8_t size = 5)** Constructor, defines the number of elements it can hold.
|
2023-06-13 19:16:10 +02:00
|
|
|
Default number of elements is 5. If **size** < 3 it will be set to 3.
|
2023-06-17 18:01:24 +02:00
|
|
|
The maximum size is currently 255, or less if there is not enough memory.
|
2023-06-13 19:16:10 +02:00
|
|
|
- **uint8_t count()** returns the number of elements in the internal array. 0.. size.
|
2023-05-18 21:21:08 +02:00
|
|
|
- **uint8_t size()** returns the maximum number of elements in the internal array.
|
2023-06-13 19:16:10 +02:00
|
|
|
- **void reset()** reset the internal counter to 0, logical clearing of the system.
|
2023-06-17 18:01:24 +02:00
|
|
|
- **bool add(float value)** add a value to the TOPMAX object if it is in the top N of maxima.
|
|
|
|
If so the smallest element is removed.
|
|
|
|
Returns false if not added or if there was an allocation error.
|
2023-06-13 19:16:10 +02:00
|
|
|
- **float getValue(uint8_t index)** get an element of the internal array.
|
|
|
|
The index must be <= **count()**, if not the function currently returns **NaN**.
|
|
|
|
This may or may not be a valid value, so the user should guard the **index** parameter carefully.
|
2023-05-18 21:21:08 +02:00
|
|
|
- **bool fill(float value)** convenience function to fill the internal array
|
2023-06-17 18:01:24 +02:00
|
|
|
with a single value e.g. 0. Returns true on success.
|
2023-05-18 21:21:08 +02:00
|
|
|
|
|
|
|
|
2023-06-13 19:16:10 +02:00
|
|
|
#### TOPMAXext
|
|
|
|
|
|
|
|
Derived from TOPMAX, extended with a tag field.
|
|
|
|
|
|
|
|
- **TOPMAXext(uint8_t size = 5)** Constructor, defines the number of elements it can hold.
|
|
|
|
Default number of elements is 5. If **size** < 3 it will be set to 3.
|
2023-06-17 18:01:24 +02:00
|
|
|
The maximum size is currently 255, or less if there is not enough memory.
|
2023-06-13 19:16:10 +02:00
|
|
|
- **bool add(float value, uint32_t tag)** add a value to the TOPMAXext object to check if
|
|
|
|
it needs to be in the top N of maxima. If so add the 32-bit **tag** too (at same index).
|
|
|
|
The 32-bit **tag** is typical an index, counter or timestamp, but any semantic is possible.
|
|
|
|
- **uint32_t getTag(uint8_t index)** get the tag from an element of the internal array.
|
|
|
|
The index must be <= **count()**, if not the function currently returns **0xFFFFFFFF**.
|
|
|
|
This may or may not be a valid value, so the user should guard the **index** parameter carefully.
|
|
|
|
- **bool fill(float value, uint32_t tag)** convenience function to fill the internal array
|
|
|
|
with a single value e.g. 0. (tag idem).
|
|
|
|
|
|
|
|
|
2023-05-18 21:21:08 +02:00
|
|
|
# Future
|
|
|
|
|
|
|
|
#### Must
|
|
|
|
|
2023-06-17 18:01:24 +02:00
|
|
|
- keep TOPMIN and TOPMAX functional in sync.
|
2023-05-18 21:21:08 +02:00
|
|
|
- improve documentation.
|
|
|
|
|
2023-06-17 18:01:24 +02:00
|
|
|
|
2023-05-18 21:21:08 +02:00
|
|
|
#### Should
|
|
|
|
|
|
|
|
- add unit tests.
|
2023-06-17 18:01:24 +02:00
|
|
|
- for extended version if possible.
|
|
|
|
|
2023-05-18 21:21:08 +02:00
|
|
|
|
|
|
|
#### Could
|
|
|
|
|
2023-06-17 18:01:24 +02:00
|
|
|
- add more examples.
|
|
|
|
- example creative use of tag field.
|
|
|
|
- add error handling
|
|
|
|
- TOP_ERR_ALLOCATION
|
|
|
|
- TOP_ERR_INDEX
|
|
|
|
- TOP_NOT_ADDED
|
|
|
|
- TOP_OK
|
|
|
|
- int error; **int lastError()**
|
2023-06-13 19:16:10 +02:00
|
|
|
- create template class.
|
2023-06-17 18:01:24 +02:00
|
|
|
- **bool check(float value)** and **bool check(float value, uint32_t tag)**
|
|
|
|
- if (value < getValue(0)) or so.
|
|
|
|
- **inRange(value)**
|
|
|
|
- checks if this value would be added to the TOP-N
|
|
|
|
- **TOP** class, in which the condition can be set as parameter.
|
|
|
|
- a function() returning true or false when comparing 2 values.
|
|
|
|
- **bool hasValue(float value)** and **bool hasTag(uint32_t tag)**
|
|
|
|
- or **int getIndex(...)** duplicates?
|
|
|
|
|
2023-05-18 21:21:08 +02:00
|
|
|
|
|
|
|
#### Wont
|
|
|
|
|
2023-06-17 18:01:24 +02:00
|
|
|
- how to handle double/triple etc. entries with same value
|
|
|
|
- they are handled as unique elements, that is what I had in mind.
|
|
|
|
- optimize loops in **fill()** (maybe upon request)
|
|
|
|
- pointer math
|
|
|
|
|