Contents
- Overview
- Requirements
- Design and Rationale
- Building
- Examples
Overview
This library consists of a set of templated functions for doing basic arithmetic operations on containers and ranges of iterators. The elements of the ranges/containers can be any built in aritmetic type, e.g. float
, double
, int
, or any other type that supports the arithmetic operations: +, -, *, /. We refer to a range/container of arithmetic elements as a mathematical vector. This library defines vector operations for arbitrary finite vector spaces. This library does not define any specific data structure for the vectors, but operates on general vectors/ranges/containers.
The library consists of different modules:
- Vector Spaces. This module defines elementwise arithmetic operations on vectors. It consists of the functions: add, subtract, negate, multiply, divide.
- Normed and Metric Spaces. This module defines norms/lengths and metrics/distances for vectors. These functions take one or two vectors and returns a single scalar. It consists of the functions:
dot
, norm
, distance
, squared_norm
, squared_distance
. They are defined for the following vector spaces: Euclidean Space (L-2), Manhattan Space (L-1), Maximum Space (L-Infinity).
- Misc Operations. This contains the functions:
sum
, convert
.
- Logical Operations. This module defines elementwise boolean operations on ranges/containers. The elements should be of type
bool
, or any other type that supports the the boolean operations &&, ||, !. The module contains the functions: logical_and, logical_or, logical_not.
- STD Algorithms on Containers. This module defines container versions of some range algorithms from the standard library header algorithm. It only does it for the algorithms that are used a lot for arithmetic types. It contains the functions:
fill
, copy
, min_element
, max_element
, minmax_element
.
Requirements
All the functions of this library assume that the input and output is:
- Either ranges of iterators, or standard like containers, i.e. they have begin and end functions that give iterators.
- The elements of the ranges/containers should support arithmetic operations: +, -, *, /. The functions in the module Logical Operations is the only exception to this. It assumes that the elements support the boolean operations &&, ||, !.
- The functions in the norms and metrics module assume that the default construction of a scalar gives zero, which is true for the built-in arithmetic types. The function
sum
also assumes this.
Design and Rationale
This library deals with arithmetic operations on mathematical vectors, represented as ranges or containers. It is designed for vectors in arbitrary dimensions. It is NOT optimized for small vectors, which you typically use to represent geometry in 2D and 3D. For that use case it might be more optimal to use other libraries like: GLM and Eigen.
C++ libraries that deals with mathematical vectors in arbitrary dimensions can usually be divided into two categories:
- Those that implement a vector class and overload its arithmetic operators. Examples of this are: std::valarray, Eigen, Blitz++, OpenCV.
- Those that implement general algorithms that work on any container or range of iterators. An example of this is the standard library and the headers algorithm and numeric.
This library follows the second approach. It implements some common arithmetic algorithms that are missing in the standard library.
The two approaches have different advantages and disadvantages:
- The advantage of the first approach is that the syntax can be very concise, when you overload the arithmetic operators to act directly on the vector class. This approach can also have performance advantages, since it couples the algorithm to the data structure. It is then possible to use techniques like expression templates or aligned memory and vectorization.
- The advantage of the second approach is that it allows generic programming. Code that uses such libraries is more general since it is less coupled to a specific data structure. This is good for readability and reuse of the code.
Building
The library is header only, which makes it easy to use and build. Download it and add it to the include directory of your project and #include <aaa.hpp>
.
Examples
The algorithms can be used on arbitrary containers like this:
Image blend(const Image& in1, const Image& in2)
{
auto a = add(in1, in2);
return divide(a, 2);
}
std::vector<float> project(const std::vector<float>& a, const std::vector<float>& b)
{
return multiply(scaling, b);
}