2023-05-18 21:22:02 +02:00
|
|
|
|
|
|
|
[![Arduino CI](https://github.com/RobTillaart/TOPMIN/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
|
|
|
|
[![Arduino-lint](https://github.com/RobTillaart/TOPMIN/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/TOPMIN/actions/workflows/arduino-lint.yml)
|
|
|
|
[![JSON check](https://github.com/RobTillaart/TOPMIN/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/TOPMIN/actions/workflows/jsoncheck.yml)
|
|
|
|
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/TOPMIN/blob/master/LICENSE)
|
|
|
|
[![GitHub release](https://img.shields.io/github/release/RobTillaart/TOPMIN.svg?maxAge=3600)](https://github.com/RobTillaart/TOPMIN/releases)
|
|
|
|
|
|
|
|
|
|
|
|
# TOPMIN
|
|
|
|
|
|
|
|
Arduino library to track the top N minima.
|
|
|
|
|
|
|
|
## Description
|
|
|
|
|
|
|
|
This experimental library tracks the top N minima of a series of values.
|
2023-06-16 09:46:29 +02:00
|
|
|
|
|
|
|
The library implements two classes:
|
|
|
|
- **TOPMIN** just tracks the values. (smaller footprint).
|
|
|
|
- **TOPMINext** 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 minima temperature / humidity over a given period of time.
|
|
|
|
- track top N UV minima intensity during the day.
|
|
|
|
- track top N trading volumes or $$.
|
|
|
|
- track top N smallest fish caught in a contest.
|
|
|
|
- track top N minimum number of visitors during a day.
|
|
|
|
|
|
|
|
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:22:02 +02:00
|
|
|
|
|
|
|
|
2023-06-18 10:25:40 +02:00
|
|
|
#### 0.2.0 breaking change
|
|
|
|
|
|
|
|
Since version 0.2.0 the minima is found with **getValue(index == 0)**.
|
|
|
|
In earlier versions the minima was found with **index == count()**.
|
|
|
|
|
|
|
|
The advantage of this change is that independent of the size of the TOPMIN
|
|
|
|
object the true minima will always at index == 0. So if you change your code
|
|
|
|
and the size of the TOPMIN object, far less code needs to be changed.
|
|
|
|
It also allows to have two or more TOPMIN objects of different size and use one
|
|
|
|
index to access both.
|
|
|
|
|
|
|
|
|
2023-05-18 21:22:02 +02:00
|
|
|
#### Links
|
|
|
|
|
2023-06-16 09:46:29 +02:00
|
|
|
- https://github.com/RobTillaart/TOPMAX
|
2023-05-18 21:22:02 +02:00
|
|
|
- https://github.com/RobTillaart/TOPMIN
|
|
|
|
- https://github.com/RobTillaart/runningAverage
|
|
|
|
- https://github.com/RobTillaart/MINMAX
|
|
|
|
|
|
|
|
|
|
|
|
## Interface
|
|
|
|
|
|
|
|
```cpp
|
|
|
|
#include "TOPMIN.h"
|
|
|
|
```
|
|
|
|
|
2023-06-16 09:46:29 +02:00
|
|
|
#### TOPMIN
|
|
|
|
|
2023-05-18 21:22:02 +02:00
|
|
|
- **TOPMIN(uint8_t size = 5)** Constructor, defines the number of elements it can hold.
|
2023-06-16 09:46:29 +02:00
|
|
|
Default number of elements is 5. If **size** < 3 it will be set to 3.
|
2023-06-18 10:25:40 +02:00
|
|
|
The maximum size is currently 255, or less if there is not enough memory.
|
2023-06-16 09:46:29 +02:00
|
|
|
- **uint8_t count()** returns the number of elements in the internal array. 0.. size.
|
2023-05-18 21:22:02 +02:00
|
|
|
- **uint8_t size()** returns the maximum number of elements in the internal array.
|
2023-06-16 09:46:29 +02:00
|
|
|
- **void reset()** reset the internal counter to 0, logical clearing of the system.
|
2023-06-18 10:25:40 +02:00
|
|
|
- **bool add(float value)** add a value to the TOPMIN object if it is in the top N of minima.
|
|
|
|
If so the largest element is removed.
|
|
|
|
Returns false if not added or if there was an allocation error.
|
2023-06-16 09:46:29 +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:22:02 +02:00
|
|
|
- **bool fill(float value)** convenience function to fill the internal array
|
2023-06-18 10:25:40 +02:00
|
|
|
with a single value e.g. 0. Returns true on success.
|
2023-05-18 21:22:02 +02:00
|
|
|
|
|
|
|
|
2023-06-16 09:46:29 +02:00
|
|
|
#### TOPMINext
|
|
|
|
|
|
|
|
Derived from TOPMIN, extended with a tag field.
|
|
|
|
|
|
|
|
- **TOPMINext(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-18 10:25:40 +02:00
|
|
|
The maximum size is currently 255, or less if there is not enough memory.
|
2023-06-16 09:46:29 +02:00
|
|
|
- **bool add(float value, uint32_t tag)** add a value to the TOPMINext object to check if
|
|
|
|
it needs to be in the top N of minima. 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:22:02 +02:00
|
|
|
# Future
|
|
|
|
|
|
|
|
#### Must
|
|
|
|
|
2023-06-18 10:25:40 +02:00
|
|
|
- keep TOPMIN and TOPMAX functional in sync.
|
2023-06-16 09:46:29 +02:00
|
|
|
- improve documentation.
|
2023-06-18 10:25:40 +02:00
|
|
|
|
2023-05-18 21:22:02 +02:00
|
|
|
|
|
|
|
#### Should
|
|
|
|
|
|
|
|
- add unit tests.
|
2023-06-18 10:25:40 +02:00
|
|
|
- for extended version if possible.
|
|
|
|
|
2023-05-18 21:22:02 +02:00
|
|
|
|
|
|
|
#### Could
|
|
|
|
|
2023-06-18 10:25:40 +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-16 09:46:29 +02:00
|
|
|
- create template class.
|
|
|
|
- **bool check(float value)** and **bool check(float value, uint32_t tag)**
|
|
|
|
- if (value < getValue(0)) or so.
|
2023-06-18 10:25:40 +02:00
|
|
|
- **inRange(value)**
|
2023-06-16 09:46:29 +02:00
|
|
|
- checks if this value would be added to the TOP-N
|
|
|
|
- **TOP** class, in which the condition can be set as parameter.
|
2023-06-18 10:25:40 +02:00
|
|
|
- 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:22:02 +02:00
|
|
|
|
|
|
|
|
|
|
|
#### Wont
|
|
|
|
|
2023-06-16 09:46:29 +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.
|
2023-06-18 10:25:40 +02:00
|
|
|
- optimize loops in **fill()** (maybe upon request)
|
|
|
|
- pointer math
|
|
|
|
|