2023-09-25 20:41:00 +02:00

6.5 KiB

Arduino CI Arduino-lint JSON check GitHub issues

License: MIT GitHub release PlatformIO Registry

DistanceTable

Arduino library to store a symmetrical distance table in less memory.

Description

The DistanceTable library is a class that stores a symmetrical distance table which is typically N x N entries in less memory space. It uses (N x (N-1))/2 ("in a triangle") as an euclidean distance table is symmetrical around its main diagonal. Furthermore as the main diagonal are all zero it does not need to be stored either.

An ATMEL328 (Arduino) can store a 30 x 30 matrix = 900 floats in 1740 bytes, where it would take 900 x 4 = 3600 bytes. Within the 2K RAM of an Arduino one could store normally a 21 x 21 matrix (1764 bytes). The class therefore saves around 50% of RAM. The price is performance as it takes more time to access the elements.

Interface

#include "DistanceTable.h"

Constructor set get

  • DistanceTable(uint8_t size, float value = 0.0) Constructor, allocates memory and sets initial value to all elements. If memory cannot be allocated, the size will be set to 0.
  • ~DistanceTable() Destructor, frees allocated memory.
  • void clear() sets all entries to 0.0.
  • void setAll(float value) sets all entries to value.
  • bool set(uint8_t x, uint8_t y, float value ) sets a value for (x,y) and automatically for (y, x). Checks dimensions of x and y and if x == y value should be zero. Returns true on success.
  • float get(uint8_t x, uint8_t y) gets a value from (x, y). If x == y it will return 0.

Statistics

  • float minimum(uint8_t &x, uint8_t &y) Returns the minimum and first occurrence in x and y. It skips all x == y pairs as these are 0.
  • float maximum(uint8_t &x, uint8_t &y) Returns the maximum and first occurrence in x and y. It skips all x == y pairs as these are 0.
  • float sum() Returns the sum of the stored distances times 2 (== all distances) It ignores the invert flag, as invert would make the sum 0.
  • float average() Returns the average distance (== sum() / elements() ). It ignores the invert flag, as invert would make the average 0. It skips all x == y pairs as these are 0.

Count

  • uint16_t count(float value, float epsilon = 0) counts the number of occurrences of value. As we are comparing floats the epsilon can set a margin for 'almost equal'. Default this is 0.
  • uint16_t countAbove(float value) counts the number of occurrences larger than value.
  • uint16_t countBelow(float value) counts the number of occurrences smaller than value.

Note 1: these 3 numbers do not always add up exactly as count() uses an epsilon where the other two do not. Note 2: the functions do not count the x == y diagonal. So if you want to count the values == 0.0 you should add dimension(). Behaviour might change in the future.

Invert (experimental)

  • void setInvert(bool invert = false) Set the invert flag.
  • bool getInvert() returns the current value of the invert flag.

Normally the table is value symmetrical around the y == x diagonal. With the invert flag, all values are inverted ==> get(x, y) == -get(y, x). This allows other uses for the table for only a few bytes of RAM. The table becomes a magnitude symmetrical distance table.

The performance of the distance table is affected by the invert flag as most functions need to test it. In practice this means slower, especially when the flag is set to true.

Some application thoughts:

  • difference in height between points on a route or map.
  • energy levels e.g. transitions of electrons, other physics phenomena.
  • transactions between players of a game, + for one is - for the other.
  • hold a 2D colour space

Note: this functionality is not tested extensively.

Debug

  • void dump(Print * stream = &Serial) dumps complete distance table, default to Serial. Note in previous releases it dumped only a "triangle".
  • uint8_t dimension() dimension of the table == parameter in constructor.
  • uint16_t elements() amount of elements allocated.
  • uint16_t memoryUsed() amount of memory used. Typical elements() x sizeof(float).

Operational

See examples.

Future

Must

  • improve documentation

Should

  • improve unit tests
  • rethink class hierarchy (see ideas below).

Could

  • add examples
    • Note: table can be used for other symmetrical 2D tables.
    • And therefore include negative values
  • investigate behaviour of count() functions a bit more.
    • include diagonal?
  • could the "non-inverted" distance table be a derived or base class?
    • more performant
  • diagonal not zero class,
    • uses more bytes but allows also extra functionality while staying symmetrical.
  • what if dimension(x) = 2x dimension(y) - non square tables.
    • is there a spectrum of sparsity of a matrix?
  • int64_t int32_t int16_t int8_t + unsigned variants? (8 variations? invert flag)
    • Template class? (release 0.4.0 ?)

won't

  • clear() could set all to NAN? is that better as it indicates unknown?
    • setAll() let the user decide.
  • Hamilton paths?

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,