Skip to content

Latest commit

 

History

History
150 lines (109 loc) · 4.06 KB

README.md

File metadata and controls

150 lines (109 loc) · 4.06 KB

A simple all-template matrix class for C++

This project is useful when you just need to do some simple matrix computations in C++, and you don't want to add a complex linear algebra framework as a dependency.

Pros:

  • Header-only (actually, everything is in a single header).
  • No dynamic memory allocation for the matrix data.
  • Compile-time dimensions checking for all matrix operations.
  • Supports operations on mixed types, such as the sum of a matrix of int and a matrix of float.
  • Modern C++ with no external dependencies.
  • Suited for embedded/microcontroller environments (C++11 support needed, though).

Cons:

  • Not optimized for large matrices. Consider using Armadillo or Eigen in this case.

Declare your matrix types

The matrix.h header provides the template MatrixBase<T,R,C> class. It is recommended to explicitly define the matrix types you need based on this class.

#include "matrix.h"

using Matrix2f  = MatrixBase<float,2,2>; // 2x2 square matrix
using Matrix3f  = MatrixBase<float,3,3>; // 3x3 square matrix
using Matrix32f = MatrixBase<float,3,2>; // 3x2 matrix (3 rows, 2 columns)
using Matrix23f = MatrixBase<float,2,3>; // 2x3 matrix (2 rows, 3 columns)
using RVector3f = MatrixBase<float,1,3>; // 1x3 row vector
using CVector3f = MatrixBase<float,3,1>; // 3x1 column vector
using Scalarf   = MatrixBase<float,1,1>; // 1x1 scalar

Constructors

The default constructor leaves the matrix unintialized. This is done for performance reasons when you declare a temporary matrix and fill it element-wise.

To construct a matrix where each element has the same value, just pass the value to the constructor.

/*
 * myMatrix =
 * 0 0 0
 * 0 0 0
 * 0 0 0
 */
Matrix3f myMatrix(0);

Initializer-lists are used to initialize a matrix to an arbitrary value.

Matrix3f myMatrix(
{
    1, 2, 3,
    4, 5, 6,
    7, 8, 9
});

The eye() static member function produces the identity matrix.

/*
 * I =
 * 1 0 0
 * 0 1 0
 * 0 0 1
 */
auto I = Matrix3f::eye();

Accessors

rows() and cols() return the matrix dimensions. size() returning a tuple can be used instead.

unsigned int rows = myMatrix.rows();
unsigned int cols = myMatrix.cols();
auto sz = myMatrix.size();
assert(rows == get<0>(sz));
assert(cols == get<1>(sz));

Individual elements of the matrix can be accessed using either operator() or at(). operator() does not perform bound checking, while at() throws std::range_error if the access is out of bounds.

float f = myMatrix(0,0);
myMatrix(0,0) = 15;

auto x = myMatrix(100,100);    // Undefined behavior
auto y = myMatrix.at(100,100); // throws std::range_error

A matrix can be printed with operator<< on an ostream.

cout << myMatrix << endl;

Operations

Operator overloading provides matrix operations.

Matrix32f a(0), b(0), c(0);

Matrix23f d = transpose(a);

c = a + b;
c = a + 1;

c += a;
c += 1;

c = a - b;
c = a - 1;

c -= a;
c -= 1;

Matrix3f e = a * d;
c = a * 2;

e *= Matrix2f(2);
c = a * 2;

c = a / 2;
c /= 2;

Matrix inverse and determinant are supported only for matrices up to 4x4. This is done on purpose: 4x4 is about the limit where complex algorithms are needed or performance drops significantly. If you need to invert large matrices, it probably means you have outgrown this simple matrix library and need to switch to a full-fledged linear algebra library. Consider using Armadillo or Eigen.

There are two overloads of inv(). The one taking as argument only the matrix to invert throws std::runtime_error if the matrix is singular, while the one taking the matrix determinant as second parameter produces undefined behavior.

// Throws std::runtime_error if matrix singular
auto g = inv(myMatrix);

// Split determinant computation and inverse, allows explicit singularity check
const float threshold = 1e-3f;
float determinant = det(myMatrix);
if(fabs(determinant) < threshold) cerr << "Ill-conditioned matrix" << endl;
else {
    auto g = inv(myMatrix, determinant);
}