summaryrefslogtreecommitdiffstats
path: root/include/mmmatrix.hpp
diff options
context:
space:
mode:
Diffstat (limited to 'include/mmmatrix.hpp')
-rw-r--r--include/mmmatrix.hpp148
1 files changed, 98 insertions, 50 deletions
diff --git a/include/mmmatrix.hpp b/include/mmmatrix.hpp
index d18f3f1..b973aba 100644
--- a/include/mmmatrix.hpp
+++ b/include/mmmatrix.hpp
@@ -13,12 +13,13 @@
#include <iostream>
#include <cstring>
+#include <cassert>
+#include <initializer_list>
namespace mm {
template<typename T, std::size_t Rows, std::size_t Cols>
class basic_matrix;
-
template<typename T, std::size_t Rows, std::size_t Cols>
class matrix;
@@ -43,73 +44,102 @@ public:
static constexpr std::size_t rows = Rows;
static constexpr std::size_t cols = Cols;
+ // from initializer_list
+ basic_matrix(std::initializer_list<std::initializer_list<T>> l);
+
+ // copyable and movable
basic_matrix(const basic_matrix<T, Rows, Cols>& other);
basic_matrix(basic_matrix<T, Rows, Cols>&& other);
+ // copy from another matrix
template<std::size_t ORows, std::size_t OCols>
basic_matrix(const basic_matrix<T, ORows, OCols>& other);
+ // copy or move from 2D array
basic_matrix(const T (& values)[Rows][Cols]);
basic_matrix(T (&& values)[Rows][Cols]);
// access data
T& at(std::size_t row, std::size_t col);
+ const T& at(std::size_t row, std::size_t col) const;
auto&& operator[](std::size_t index);
void swap_rows(std::size_t x, std::size_t y);
void swap_cols(std::size_t x, std::size_t y);
// mathematical operations
- basic_matrix<T, Cols, Rows> transposed();
- inline basic_matrix<T, Cols, Rows> trd() { return transposed(); }
+ basic_matrix<T, Cols, Rows> transposed() const;
+ inline basic_matrix<T, Cols, Rows> trd() const { return transposed(); }
- // bool is_invertible();
- // bool invert();
- // basic_matrix<T, Rows, Cols> inverse();
+ // bool is_invertible() const;
+ // basic_matrix<T, Rows, Cols> inverse() const;
/// downcast to square matrix
- inline constexpr bool is_square() { return (Rows == Cols); }
- inline constexpr square_matrix<T, Rows> to_square() {
+ inline constexpr bool is_square() const { return (Rows == Cols); }
+ inline constexpr square_matrix<T, Rows> to_square() const {
static_assert(is_square());
return static_cast<square_matrix<T, Rows>>(*this);
}
/// downcast to row_vector
- inline constexpr bool is_row_vec() { return (Cols == 1); }
- inline constexpr row_vec<T, Rows> to_row_vec() {
+ inline constexpr bool is_row_vec() const { return (Cols == 1); }
+ inline constexpr row_vec<T, Rows> to_row_vec() const {
static_assert(is_row_vec());
return static_cast<row_vec<T, Rows>>(*this);
}
/// downcast to col_vector
- inline constexpr bool is_col_vec() { return (Rows == 1); }
- inline constexpr col_vec<T, Cols> to_col_vec() {
+ inline constexpr bool is_col_vec() const { return (Rows == 1); }
+ inline constexpr col_vec<T, Cols> to_col_vec() const {
static_assert(is_col_vec());
return static_cast<col_vec<T, Cols>>(*this);
}
private:
- T data[Rows][Cols] = {};
+ T data[Rows * Cols] = {};
};
template<typename T, std::size_t Rows, std::size_t Cols>
-mm::basic_matrix<T, Rows, Cols>::basic_matrix(const mm::basic_matrix<T, Rows, Cols>& other) {
- for (int row = 0; row < Rows; row++)
- for (int col = 0; col < Cols; col++)
- data[row][col] = other.data[row][col];
+mm::basic_matrix<T, Rows, Cols>::basic_matrix(
+ std::initializer_list<std::initializer_list<T>> l
+) {
+ assert(l.size() == Rows);
+
+ auto row_it = l.begin();
+ for (unsigned row = 0; (row < Rows) && (row_it != l.end()); row++) {
+ assert((*row_it).size() == Cols);
+
+ auto col_it = (*row_it).begin();
+ for (unsigned col = 0; (col < Cols) && (col_it != (*row_it).end()); col++) {
+ this->at(row, col) = *col_it;
+ ++col_it;
+ }
+ ++row_it;
+ }
+}
+
+template<typename T, std::size_t Rows, std::size_t Cols>
+mm::basic_matrix<T, Rows, Cols>::basic_matrix(
+ const mm::basic_matrix<T, Rows, Cols>& other
+) {
+ std::memcpy(&data, &other.data, sizeof(data));
}
template<typename T, std::size_t Rows, std::size_t Cols>
-mm::basic_matrix<T, Rows, Cols>::basic_matrix(mm::basic_matrix<T, Rows, Cols>&& other) {
+mm::basic_matrix<T, Rows, Cols>::basic_matrix(
+ mm::basic_matrix<T, Rows, Cols>&& other
+) {
data = other.data;
}
template<typename T, std::size_t Rows, std::size_t Cols>
template<std::size_t ORows, std::size_t OCols>
-mm::basic_matrix<T, Rows, Cols>::basic_matrix(const mm::basic_matrix<T, ORows, OCols>& other) {
+mm::basic_matrix<T, Rows, Cols>::basic_matrix(
+ const mm::basic_matrix<T, ORows, OCols>& other
+) {
static_assert((ORows <= Rows),
"cannot copy a taller matrix into a smaller one"
);
@@ -118,9 +148,7 @@ mm::basic_matrix<T, Rows, Cols>::basic_matrix(const mm::basic_matrix<T, ORows, O
"cannot copy a larger matrix into a smaller one"
);
- for (int row = 0; row < Rows; row++)
- for (int col = 0; col < Cols; col++)
- data[row][col] = other.data[row][col];
+ std::memcpy(&data, &other.data, sizeof(data));
}
template<typename T, std::size_t Rows, std::size_t Cols>
@@ -138,20 +166,32 @@ mm::basic_matrix<T, Rows, Cols>::basic_matrix(T (&& values)[Rows][Cols]) {
template<typename T, std::size_t Rows, std::size_t Cols>
T& mm::basic_matrix<T, Rows, Cols>::at(std::size_t row, std::size_t col) {
- static_assert(row < Rows, "out of row bound");
- static_assert(col < Cols, "out of column bound");
+ assert(row < Rows); // "out of row bound"
+ assert(col < Cols); // "out of column bound"
+
+ return data[row * Cols + col];
+}
+
+template<typename T, std::size_t Rows, std::size_t Cols>
+const T& mm::basic_matrix<T, Rows, Cols>::at(std::size_t row, std::size_t col) const {
+ assert(row < Rows); // "out of row bound"
+ assert(col < Cols); // "out of column bound"
- return data[row][col];
+ return data[row * Cols + col];
}
template<typename T, std::size_t Rows, std::size_t Cols>
auto&& mm::basic_matrix<T, Rows, Cols>::operator[](std::size_t index) {
- if constexpr (is_row_vec())
- return data[0][index];
- else if constexpr (is_col_vec())
- return data[index][0];
+ if constexpr (is_row_vec()) {
+ static_assert(index < Rows);
+ return data[index];
+ } else if constexpr (is_col_vec()) {
+ static_assert(index < Cols);
+ return data[index];
+ }
- return row_vec<T, Rows>(std::move(data[index]));
+ // TODO: fix
+ // return row_vec<T, Rows>(std::move(data[index]));
}
template<typename T, std::size_t Rows, std::size_t Cols>
@@ -159,8 +199,8 @@ void mm::basic_matrix<T, Rows, Cols>::swap_rows(std::size_t x, std::size_t y) {
if (x == y)
return;
- for (int col = 0; col < Cols; col++)
- std::swap(data[x][col], data[y][col]);
+ for (unsigned col = 0; col < Cols; col++)
+ std::swap(this->at(x, col), this->at(y, col));
}
template<typename T, std::size_t Rows, std::size_t Cols>
@@ -168,17 +208,17 @@ void mm::basic_matrix<T, Rows, Cols>::swap_cols(std::size_t x, std::size_t y) {
if (x == y)
return;
- for (int row = 0; row < rows; row++)
- std::swap(data[row][x], data[row][y]);
+ for (unsigned row = 0; row < rows; row++)
+ std::swap(this->at(row, x), this->at(row, y));
}
template<typename T, std::size_t M, std::size_t N>
-mm::basic_matrix<T, N, M> mm::basic_matrix<T, M, N>::transposed() {
+mm::basic_matrix<T, N, M> mm::basic_matrix<T, M, N>::transposed() const {
mm::basic_matrix<T, N, M> result;
- for (int row = 0; row < M; row++)
- for (int col = 0; col < N; col++)
- result[row][col] = this[col][row];
+ for (unsigned row = 0; row < M; row++)
+ for (unsigned col = 0; col < N; col++)
+ result.at(row, col) = this->at(row, col);
return result;
}
@@ -192,8 +232,8 @@ mm::basic_matrix<T, Rows, Cols> operator+(
) {
mm::basic_matrix<T, Rows, Cols> result;
- for (int row = 0; row < Rows; row++)
- for (int col = 0; col < Cols; col++)
+ for (unsigned row = 0; row < Rows; row++)
+ for (unsigned col = 0; col < Cols; col++)
result.at(row, col) = a.at(row, col) + a.at(row, col);
return result;
@@ -205,8 +245,8 @@ mm::basic_matrix<T, Rows, Cols> operator*(
const T& scalar
) {
mm::basic_matrix<T, Rows, Cols> result;
- for (int row = 0; row < Rows; row++)
- for (int col = 0; col < Cols; col++)
+ for (unsigned row = 0; row < Rows; row++)
+ for (unsigned col = 0; col < Cols; col++)
result.at(row, col) = m.at(row, col) * scalar;
return result;
@@ -228,8 +268,8 @@ mm::basic_matrix<T, M, N> operator*(
mm::basic_matrix<T, M, N> result;
// TODO: use a more efficient algorithm
- for (int row = 0; row < M; row++)
- for (int col = 0; col < N; col++)
+ for (unsigned row = 0; row < M; row++)
+ for (unsigned col = 0; col < N; col++)
for (int k = 0; k < P; k++)
result.at(row, col) = a.at(row, k) * b.at(k, col);
@@ -247,12 +287,12 @@ mm::basic_matrix<T, Rows, Cols> operator-(
template<typename T, std::size_t Rows, std::size_t Cols>
std::ostream& operator<<(std::ostream& os, const mm::basic_matrix<T, Rows, Cols>& m) {
- for (int row = 0; row < Rows; row++) {
+ for (unsigned row = 0; row < Rows; row++) {
os << "[ ";
- for (int col = 0; col < (Cols -1); col++) {
- os << m.at(row, col);
+ for (unsigned col = 0; col < (Cols -1); col++) {
+ os << m.at(row, col) << ", ";
}
- os << m.at(Rows -1, Cols -1) << " ]\n";
+ os << m.at(row, (Cols -1)) << " ]\n";
}
return os;
@@ -275,8 +315,8 @@ public:
template<typename T, std::size_t N>
void mm::square_matrix<T, N>::transpose() {
- for (int row = 0; row < N; row++)
- for (int col = 0; col < row; col++)
+ for (unsigned row = 0; row < N; row++)
+ for (unsigned col = 0; col < row; col++)
std::swap(this->at(row, col), this->at(col, row));
}
@@ -285,10 +325,18 @@ void mm::square_matrix<T, N>::transpose() {
template<typename T, std::size_t Rows>
class mm::row_vec : public mm::basic_matrix<T, Rows, 1> {
public:
+ using mm::basic_matrix<T, Rows, 1>::basic_matrix;
};
/* column vector specialization */
template<typename T, std::size_t Cols>
class mm::col_vec : public mm::basic_matrix<T, 1, Cols> {
public:
+ using mm::basic_matrix<T, 1, Cols>::basic_matrix;
+};
+
+template<typename T, std::size_t Rows, std::size_t Cols>
+class mm::matrix : public mm::basic_matrix<T, Rows, Cols> {
+public:
+ using mm::basic_matrix<T, Rows, Cols>::basic_matrix;
};