summaryrefslogtreecommitdiffstats
path: root/include/mm/mmmatrix.hpp
diff options
context:
space:
mode:
authorancarola <raffaele.ancarola@epfl.ch>2019-06-30 22:21:54 +0200
committerancarola <raffaele.ancarola@epfl.ch>2019-06-30 22:21:54 +0200
commit6e39d531ed36d043e6f6fee6befca2be00fd3f57 (patch)
tree9a1dad6301b5dde13bb64bfc7898b1922c6f2c5c /include/mm/mmmatrix.hpp
parentFinally compiles (diff)
downloadlibmm-6e39d531ed36d043e6f6fee6befca2be00fd3f57.tar.gz
libmm-6e39d531ed36d043e6f6fee6befca2be00fd3f57.zip
Optimized matrix section
- Vector iterators: allow to iterate on rows, columns or diagonals - Transposition doesn't affect allocated space, O(1)
Diffstat (limited to 'include/mm/mmmatrix.hpp')
-rw-r--r--include/mm/mmmatrix.hpp716
1 files changed, 262 insertions, 454 deletions
diff --git a/include/mm/mmmatrix.hpp b/include/mm/mmmatrix.hpp
index 9910273..f88e90c 100644
--- a/include/mm/mmmatrix.hpp
+++ b/include/mm/mmmatrix.hpp
@@ -17,39 +17,20 @@
#include <cassert>
#include <initializer_list>
#include <array>
+#include <memory>
+
+#include "mm/mmiterator.hpp"
namespace mm {
template<typename T, std::size_t Rows, std::size_t Cols>
class basic_matrix;
- /* specialization of basic_matrix for Rows == Cols */
- template<typename T, std::size_t N>
- class square_interface;
-
- /*
- * Most general type of matrix iterator
- * IterType = Row, Column, Diagonal
- * Grid = constness of mm::basic_matrix
- */
- template<typename T, std::size_t Rows, std::size_t Cols, int IterType, class Grid>
- class vector_iterator;
-
- /*
- * Access methods interface
- */
-
- template<typename T, std::size_t Rows, std::size_t Cols, bool Regular>
- class access;
-
/* specialisations */
template<typename T, std::size_t Rows, std::size_t Cols>
class matrix; // simple matrix format
- template<typename T, std::size_t Rows, std::size_t Cols>
- class t_matrix; // transposed matrix format
-
/* specialization of basic_matrx for Cols = 1 */
template<typename T, std::size_t Rows>
class row_vec;
@@ -61,155 +42,20 @@ namespace mm {
template<typename T, std::size_t N>
class square_matrix;
- template<typename T, std::size_t N>
- class t_square_matrix;
-
/* specialisation of a square_matrix for a sub-diagonal composed matrix */
template<typename T, std::size_t N, std::size_t K = 0>
class diag_matrix;
-
- template<typename T, std::size_t N, std::size_t K = 0>
- class t_diag_matrix;
}
-// TODO, short term solution
-#define MM_ROW_ITER 0
-#define MM_COL_ITER 1
-#define MM_DIAG_ITER 2
-
-template<typename T, std::size_t Rows, std::size_t Cols, int IterType, class Grid>
-class mm::vector_iterator
-{
- std::size_t index; // variable index
-
- Grid& M;
-
- const int position; // fixed index, negative too for diagonal iterator
-
-public:
- template<typename U, std::size_t ORows, std::size_t OCols, class OIterType, class OGrid>
- friend class vector_iterator;
-
- vector_iterator(Grid& M, int position, std::size_t index = 0);
-
- mm::vector_iterator<T, Rows, Cols, IterType, Grid> operator++()
- {
- vector_iterator<T, Rows, Cols, IterType, Grid> it = *this;
- ++index;
- return it;
- }
-
- mm::vector_iterator<T, Rows, Cols, IterType, Grid> operator--()
- {
- vector_iterator<T, Rows, Cols, IterType, Grid> it = *this;
- --index;
- return it;
- }
-
- mm::vector_iterator<T, Rows, Cols, IterType, Grid>& operator++(int)
- {
- ++index;
- return *this;
- }
-
- mm::vector_iterator<T, Rows, Cols, IterType, Grid>& operator--(int)
- {
- --index;
- return *this;
- }
-
- bool operator==(const mm::vector_iterator<T, Rows, Cols, IterType, Grid>& other) const
- {
- return index == other.index;
- }
-
- bool operator!=(const mm::vector_iterator<T, Rows, Cols, IterType, Grid>& other) const
- {
- return index != other.index;
- }
-
- bool ok() const
- {
- if constexpr(IterType == MM_ROW_ITER)
- return index < Cols;
- else
- return index < Rows;
- }
- T& operator*() const;
- T& operator[](std::size_t);
-};
-
-/* Row Iterators */
-
-namespace mm {
-
- template<typename T, std::size_t Rows, std::size_t Cols>
- using row_iterator = vector_iterator<T, Rows, Cols, MM_ROW_ITER, mm::basic_matrix<T, Rows, Cols>>;
-
- template<typename T, std::size_t Rows, std::size_t Cols>
- using col_iterator = vector_iterator<T, Rows, Cols, MM_COL_ITER, mm::basic_matrix<T, Rows, Cols>>;
-
- template<typename T, std::size_t Rows, std::size_t Cols>
- using const_row_iterator = vector_iterator<T, Rows, Cols, MM_ROW_ITER, std::add_const<mm::basic_matrix<T, Rows, Cols>>>;
-
- template<typename T, std::size_t Rows, std::size_t Cols>
- using const_col_iterator = vector_iterator<T, Rows, Cols, MM_COL_ITER, std::add_const<mm::basic_matrix<T, Rows, Cols>>>;
+/*namespace mm {
template<typename T, std::size_t N>
using diag_iterator = vector_iterator<T, N, N, MM_DIAG_ITER, mm::basic_matrix<T, N, N>>;
template<typename T, std::size_t N>
- using const_diag_iterator = vector_iterator<T, N, N, MM_DIAG_ITER, std::add_const<mm::basic_matrix<T, N, N>>>;
-}
-
-/*
- * Accessors
- */
-template<typename T, std::size_t Rows, std::size_t Cols, bool Regular>
-class mm::access
-{
-public:
-
- //access(mm::basic_matrix<T, Rows, Cols>& ref) : M(ref) {}
-
- 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);
- auto operator[](std::size_t index) const;
-
-//private:
-// mm::basic_matrix<T, Rows, Cols>& M;
-protected:
- std::array<T, Rows * Cols> data;
-};
-
-/*
- * Square interface
- */
-
-template<typename T, std::size_t N>
-class mm::square_interface {
-public:
-
- //square_interface(mm:basic_matrix<T, N, N>& _M) : M(_M) {}
-
- T trace();
- inline T tr() { return trace(); }
-
- // TODO, determinant
-
- /// in place inverse
- // TODO, det != 0
- // TODO, use gauss jordan for invertible ones
- //void invert();, TODO, section algorithm
-
-//private:
-// mm:basic_matrix<T, N, N>& M; // one information more than mm::matrix !
-protected:
- std::array<T, N * N> data;
-};
+ using const_diag_iterator = vector_iterator<typename std::add_const<T>::type, N, N, MM_DIAG_ITER, typename std::add_const<mm::basic_matrix<T, N, N>>::type>;
+}*/
/*
@@ -217,15 +63,25 @@ protected:
*/
template<typename T, std::size_t Rows, std::size_t Cols>
-class mm::basic_matrix {
+class mm::basic_matrix
+{
public:
using type = T;
template<typename U, std::size_t ORows, std::size_t OCols>
friend class mm::basic_matrix;
- static constexpr std::size_t rows = Rows;
- static constexpr std::size_t cols = Cols;
+ template<typename U, std::size_t ORows, std::size_t OCols>
+ friend class mm::matrix;
+
+ template<typename U, std::size_t ORows, std::size_t OCols, class Grid>
+ friend class mm::iter::basic_iterator;
+
+ //template<typename U, std::size_t ORows, std::size_t OCols, class Grid>
+ //friend class mm::iter::basic_iterator<T, Rows, Cols, mm::basic_matrix<T, Rows, Cols>>;
+
+ //template<typename U, std::size_t ORows, std::size_t OCols, class Grid>
+ //friend class mm::iter::basic_iterator<typename std::add_const<T>::type, Rows, Cols, typename std::add_const<mm::basic_matrix<T, Rows, Cols>>::type>;
basic_matrix();
@@ -233,20 +89,13 @@ public:
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);
+ basic_matrix(const basic_matrix<T, Rows, Cols>& other) = default;
+ basic_matrix(basic_matrix<T, Rows, Cols>&& other) = default;
// copy from another matrix
template<std::size_t ORows, std::size_t OCols>
basic_matrix(const basic_matrix<T, ORows, OCols>& other);
- // access data, basic definition
- //virtual T& at(std::size_t row, std::size_t col);
- //virtual const T& at(std::size_t row, std::size_t col) const;
-
- // allows to access a matrix M at row j col k with M[j][k]
- //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);
@@ -254,35 +103,15 @@ public:
//virtual basic_matrix<T, Cols, Rows> transposed() const;
//inline basic_matrix<T, Cols, Rows> td() const { return transposed(); }
- /// downcast to square matrix
- static inline constexpr bool is_square() { 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
- static inline constexpr bool is_row_vec() { 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
- static inline constexpr bool is_col_vec() { 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);
- }
-
protected:
template<typename ConstIterator>
basic_matrix(ConstIterator begin, ConstIterator end);
-//private:
+private:
std::array<T, Rows * Cols> data;
};
+
template<typename T, std::size_t Rows, std::size_t Cols>
mm::basic_matrix<T, Rows, Cols>::basic_matrix() {
std::fill(data.begin(), data.end(), 0);
@@ -301,16 +130,6 @@ mm::basic_matrix<T, Rows, Cols>::basic_matrix(
}
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
-) : data(other.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
-) : data(std::forward<decltype(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
@@ -339,40 +158,6 @@ mm::basic_matrix<T, Rows, Cols>::basic_matrix(
std::copy(begin, end, data.begin());
}
-
-/* member functions */
-
-/*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) {
- 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 * 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() || is_col_vec()) {
- return data.at(index);
- } else {
- return mm::row_iterator<T, Rows, Cols>(*this, static_cast<int>(index));
-
- //return row_vec<T, Rows>(
- // data.cbegin() + (index * Cols),
- // data.cbegin() + ((index + 1) * Cols) + 1
- );
- }
-}*/
-
-
template<typename T, std::size_t Rows, std::size_t Cols>
void mm::basic_matrix<T, Rows, Cols>::swap_rows(std::size_t x, std::size_t y) {
if (x == y)
@@ -387,274 +172,297 @@ void mm::basic_matrix<T, Rows, Cols>::swap_cols(std::size_t x, std::size_t y) {
if (x == y)
return;
- for (unsigned row = 0; row < rows; row++)
+ 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() const {
- mm::basic_matrix<T, N, M> result;
+/*
+ * Matrix object
+ */
- for (unsigned row = 0; row < M; row++)
- for (unsigned col = 0; col < N; col++)
- result.at(col, row) = this->at(row, col);
+template<typename T, std::size_t Rows, std::size_t Cols>
+class mm::matrix
+{
+protected:
- return result;
-}*/
+ // shallow construction
+ matrix(std::shared_ptr<mm::basic_matrix<T, Rows, Cols>> grid = nullptr, bool tr = false) : M(grid), transposed(tr) {}
+public:
-/* TODO, operator overloading */
-/*template<typename T, std::size_t Rows, std::size_t Cols>
-mm::basic_matrix<T, Rows, Cols> operator+(
- const mm::basic_matrix<T, Rows, Cols>& a,
- const mm::basic_matrix<T, Rows, Cols>& b
-) {
- mm::basic_matrix<T, Rows, Cols> result;
+ //template<typename U, std::size_t ORows, std::size_t OCols>
+ using vec_iterator = mm::iter::basic_iterator<T, Rows, Cols, mm::basic_matrix<T, Rows, Cols>>;
- for (unsigned row = 0; row < Rows; row++)
- for (unsigned col = 0; col < Cols; col++)
- result.at(row, col) = a.at(row, col) + b.at(row, col);
+ //template<typename U, std::size_t ORows, std::size_t OCols>
+ using const_vec_iterator = mm::iter::basic_iterator<typename std::add_const<T>::type, Rows, Cols, typename std::add_const<mm::basic_matrix<T, Rows, Cols>>::type>;
- return result;
-}
+ // default zeros constructor
+ matrix() : M(std::make_shared<mm::basic_matrix<T, Rows, Cols>>()), transposed(false) {}
+
+ // from initializer_list
+ matrix(std::initializer_list<std::initializer_list<T>> l)
+ : M(std::make_shared<mm::basic_matrix<T, Rows, Cols>>(l)), transposed(false) {}
-template<typename T, std::size_t Rows, std::size_t Cols>
-mm::basic_matrix<T, Rows, Cols> operator*(
- const mm::basic_matrix<T, Rows, Cols>& m,
- const T& scalar
-) {
- mm::basic_matrix<T, Rows, Cols> result;
- for (unsigned row = 0; row < Rows; row++)
- for (unsigned col = 0; col < Cols; col++)
- result.at(row, col) = m.at(row, col) * scalar;
+ // copyable and movable
+ matrix(const matrix<T, Rows, Cols>& other) // deep copy
+ : M(std::make_shared<mm::basic_matrix<T, Rows, Cols>>(*other.M)), transposed(other.transposed) {}
- return result;
-}
+ matrix(basic_matrix<T, Rows, Cols>&& other) // move ptr
+ : M(other.M), transposed(other.transposed)
+ {
+ other.M = nullptr;
+ }
-template<typename T, std::size_t Rows, std::size_t Cols>
-mm::basic_matrix<T, Rows, Cols> operator*(
- const T& scalar,
- const mm::basic_matrix<T, Rows, Cols>& m
-) {
- return m * scalar;
-}
+ // copy from another matrix
+ /*template<std::size_t ORows, std::size_t OCols>
+ matrix(const matrix<T, ORows, OCols>& other)
+ : M(std::make_shared<mm::basic_matrix<T, Rows, Cols>(*other.M)), transposed(other.transposed) {} */
+
+ matrix<T, Rows, Cols> operator=(const basic_matrix<T, Rows, Cols>& other) // deep copy
+ {
+ *M = *other.M;
+ transposed = other.transposed;
+ }
-template<typename T, std::size_t M, std::size_t P1, std::size_t P2, std::size_t N>
-mm::basic_matrix<T, M, N> operator*(
- const mm::basic_matrix<T, M, P1>& a,
- const mm::basic_matrix<T, P2, N>& b
-) {
- static_assert(P1 == P2, "invalid matrix multiplication");
- mm::basic_matrix<T, M, N> result;
+ /*
+ * Transposition
+ */
- // TODO: use a more efficient algorithm
- for (unsigned row = 0; row < M; row++)
- for (unsigned col = 0; col < N; col++)
- for (unsigned k = 0; k < P1; k++)
- result.at(row, col) = a.at(row, k) * b.at(k, col);
+ matrix<T, Rows, Cols>& transpose()
+ {
+ transposed = !transposed;
+ return *this;
+ }
- return result;
-}
+ matrix<T, Rows, Cols> transpose() const
+ {
+ auto m = shallow_cpy();
+ m.transposed = !transposed;
+ return m;
+ }
-template<typename T, std::size_t Rows, std::size_t Cols>
-mm::basic_matrix<T, Rows, Cols> operator-(
- const mm::basic_matrix<T, Rows, Cols>& a,
- const mm::basic_matrix<T, Rows, Cols>& b
-) {
- return a + (static_cast<T>(-1) * b);
-}
+ inline matrix<T, Rows, Cols>& t()
+ {
+ return transpose();
+ }
-template<typename T, std::size_t Rows, std::size_t Cols, unsigned NumW = 3>
-std::ostream& operator<<(std::ostream& os, const mm::basic_matrix<T, Rows, Cols>& m) {
- for (unsigned row = 0; row < Rows; row++) {
- os << "[ ";
- for (unsigned col = 0; col < (Cols -1); col++) {
- os << std::setw(NumW) << m.at(row, col) << ", ";
- }
- os << std::setw(NumW) << m.at(row, (Cols -1)) << " ]\n";
+ inline matrix<T, Rows, Cols> t() const
+ {
+ return transpose();
}
- return os;
-}*/
+ // strongly transpose
+ matrix<T, Cols, Rows> transpose_cpy() const
+ {
+ matrix<T, Cols, Rows> out(); // copy
+ // TODO
+ }
-/*
- * derivated classes
- */
+ /*
+ * Pointer status
+ */
-// simple format matrix
-template<typename T, std::size_t Rows, std::size_t Cols>
-class mm::matrix : public mm::basic_matrix<T, Rows, Cols>, virtual public mm::access<T, Rows, Cols, true>
-{
-public:
- using mm::basic_matrix<T, Rows, Cols>::basic_matrix;
-};
+ bool expired() const
+ {
+ return M == nullptr;
+ }
-// transposed matrix
-template<typename T, std::size_t Rows, std::size_t Cols>
-class mm::t_matrix : public mm::basic_matrix<T, Rows, Cols>, virtual public mm::access<T, Rows, Cols, false>
-{
-public:
- using mm::basic_matrix<T, Rows, Cols>::basic_matrix;
-};
+ /*
+ * Downcasting conditions
+ */
-/* row vector specialization */
-template<typename T, std::size_t Rows>
-class mm::row_vec : public mm::matrix<T, Rows, 1> {
-public:
- using mm::matrix<T, Rows, 1>::matrix;
+ /// downcast to square matrix
+ static inline constexpr bool is_square() { return (Rows == Cols); }
+ inline constexpr square_matrix<T, Rows> to_square() const {
+ static_assert(is_square());
+ return static_cast<square_matrix<T, Rows>>(*this);
+ }
- // TODO, begin, end
-};
+ /// downcast to row_vector
+ static inline constexpr bool is_row_vec() { 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);
+ }
-/* column vector specialization */
-template<typename T, std::size_t Cols>
-class mm::col_vec : public mm::t_matrix<T, 1, Cols> {
-public:
- using mm::t_matrix<T, 1, Cols>::t_matrix;
+ /// downcast to col_vector
+ static inline constexpr bool is_col_vec() { 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);
+ }
- // TODO, begin, end
-};
+ /* Accessors */
-/* square matrix specialization */
-template<typename T, std::size_t N>
-class mm::square_matrix : public mm::matrix<T, N, N> , virtual public mm::square_interface<T, N> {
-public:
- using mm::matrix<T, N, N>::matrix;
+ T& at(std::size_t row, std::size_t col)
+ {
+ return (transposed) ? M->data[col * Cols + row] : M->data[row * Cols + col];
+ }
- // get the identity of size N
- static inline constexpr square_matrix<T, N> identity() {
- mm::square_matrix<T, N> i;
- for (unsigned row = 0; row < N; row++)
- for (unsigned col = 0; col < N; col++)
- i.at(row, col) = (row == col) ? 1 : 0;
+ const T& at(std::size_t row, std::size_t col) const
+ {
+ return (transposed) ? M->data[col * Cols + row] : M->data[row * Cols + col];
+ }
- return i;
+ std::size_t rows() const {
+ return (transposed) ? Cols : Rows;
}
-};
-template<typename T, std::size_t N>
-class mm::t_square_matrix : virtual public mm::t_matrix<T, N, N>, virtual public mm::square_interface<T, N> {
-public:
- using mm::t_matrix<T, N, N>::t_matrix;
-};
+ std::size_t cols() const {
+ return (transposed) ? Rows : Cols;
+ }
-/*
- * K-diagonal square matrix format
- * K is bounded between ]-N, N[
- */
+ mm::matrix<T, Rows, Cols>::vec_iterator operator[](std::size_t index)
+ {
+ return mm::matrix<T, Rows, Cols>::vec_iterator(*M, index, !transposed);
+ }
-/*template<typename T, std::size_t N, std::size_t K>
-class mm::diagonal_matrix : public mm::square_matrix<T, N>
-{
-public:
- using mm::square_matrix<T, N>::square_matrix;
+ mm::matrix<T, Rows, Cols>::const_vec_iterator operator[](std::size_t index) const
+ {
+ return mm::matrix<T, Rows, Cols>::const_vec_iterator(*M, index, !transposed);
+ }
+
+ /*
+ * Basic matematical operations (dimension indipendent)
+ */
- // TODO, redefine at, operator[]
- // TODO, matrix multiplication
-};*/
+ mm::matrix<T, Rows, Cols>& operator+=(const mm::matrix<T, Rows, Cols>& m) {
-/*template<typename T, std::size_t N>
-void mm::square_matrix<T, N>::transpose() {
- for (unsigned row = 0; row < N; row++)
- for (unsigned col = 0; col < row; col++)
- std::swap(this->at(row, col), this->at(col, row));
-}*/
+ for (unsigned row = 0; row < std::min(rows(), m.rows()); ++row)
+ for (unsigned col = 0; col < std::min(cols(), m.cols()); ++col)
+ at(row, col) += m.at(row, col);
+
+ return *this;
+ }
-/* Iterators implementation */
+ mm::matrix<T, Rows, Cols>& operator-=(const mm::matrix<T, Rows, Cols>& m) {
-template<typename T, std::size_t Rows, std::size_t Cols, int IterType, class Grid>
-mm::vector_iterator<T, Rows, Cols, IterType, Grid>::vector_iterator(Grid& _M, int pos, std::size_t i)
- : index(i), M(_M), position(pos)
-{
- if constexpr (IterType == MM_ROW_ITER) {
- assert(pos < Cols);
- } else if constexpr (IterType == MM_COL_ITER) {
- assert(pos < Rows);
- } else if constexpr (IterType == MM_DIAG_ITER) {
- assert(abs(pos) < Rows);
+ for (unsigned row = 0; row < std::min(rows(), m.rows()); ++row)
+ for (unsigned col = 0; col < std::min(cols(), m.cols()); ++col)
+ at(row, col) -= m.at(row, col);
+
+ return *this;
}
-}
-template<typename T, std::size_t Rows, std::size_t Cols, int IterType, class Grid>
-T& mm::vector_iterator<T, Rows, Cols, IterType, Grid>::operator*() const
-{
- if constexpr (IterType == MM_ROW_ITER)
- return M.data[position * Cols + index];
- else if constexpr (IterType == MM_COL_ITER)
- return M.data[index * Cols + position];
- else if constexpr (IterType == MM_DIAG_ITER)
- return (position > 0) ?
- M.data[(index + position) * Cols + index] :
- M.data[index * Cols + (index - position)];
-}
+ mm::matrix<T, Rows, Cols> operator*=(const T& k) {
-template<typename T, std::size_t Rows, std::size_t Cols, int IterType, class Grid>
-T& mm::vector_iterator<T, Rows, Cols, IterType, Grid>::operator[](std::size_t i)
-{
- if constexpr (IterType == MM_ROW_ITER)
- return M.data[position * Cols + i];
- else if constexpr (IterType == MM_COL_ITER)
- return M.data[i * Cols + position];
- else if constexpr (IterType == MM_DIAG_ITER)
- return (position > 0) ?
- M.data[(i + position) * Cols + i] :
- M.data[i * Cols + (i - position)];
-}
+ for (unsigned row = 0; row < rows(); ++row)
+ for (auto& x : (*this)[row])
+ x *= k;
+
+ return *this;
+ }
-/*
- * Accessors implementation
- */
+protected:
-template<typename T, std::size_t Rows, std::size_t Cols, bool Regular>
-T& mm::access<T, Rows, Cols, Regular>::at(std::size_t row, std::size_t col)
-{
- if constexpr (Regular)
- return data[row * Cols + col];
- else
- return data[col * Cols + row]; // transpose
+ std::shared_ptr<mm::basic_matrix<T, Rows, Cols>> M;
+
+ matrix<T, Rows, Cols> shallow_cpy()
+ {
+ return matrix<T, Rows, Cols>(M, transposed);
+ }
+
+private:
+
+ bool transposed;
+};
+
+/* Basic operator overloading (dimension indipendent) */
+
+template<typename T, std::size_t Rows, std::size_t Cols>
+mm::matrix<T, Rows, Cols> operator+(
+ mm::matrix<T, Rows, Cols> a,
+ const mm::matrix<T, Rows, Cols>& b
+) {
+ return a += b;
}
-template<typename T, std::size_t Rows, std::size_t Cols, bool Regular>
-const T& mm::access<T, Rows, Cols, Regular>::at(std::size_t row, std::size_t col) const
-{
- if constexpr (Regular)
- return data[row * Cols + col];
- else
- return data[col * Cols + row]; // transpose
+template<typename T, std::size_t Rows, std::size_t Cols>
+mm::matrix<T, Rows, Cols> operator-(
+ mm::matrix<T, Rows, Cols> a,
+ const mm::matrix<T, Rows, Cols>& b
+) {
+ return a -= b;
}
-template<typename T, std::size_t Rows, std::size_t Cols, bool Regular>
-auto mm::access<T, Rows, Cols, Regular>::operator[](std::size_t index)
-{
- if constexpr (this->is_row_vec() || this->is_col_vec())
- return data.at(index);
- else if (Regular)
- return mm::row_iterator<T, Rows, Cols>(*this, static_cast<int>(index));
- else
- return mm::col_iterator<T, Rows, Cols>(*this, static_cast<int>(index));
+template<typename T, std::size_t Rows, std::size_t Cols>
+mm::matrix<T, Rows, Cols> operator*(
+ mm::matrix<T, Rows, Cols> a,
+ const T& k
+) {
+ return a *= k;
}
-template<typename T, std::size_t Rows, std::size_t Cols, bool Regular>
-auto mm::access<T, Rows, Cols, Regular>::operator[](std::size_t index) const
-{
- if constexpr (this->is_row_vec() || this->is_col_vec())
- return data.at(index);
- else if (Regular)
- return mm::const_row_iterator<T, Rows, Cols>(*this, static_cast<int>(index));
- else
- return mm::const_col_iterator<T, Rows, Cols>(*this, static_cast<int>(index));
+template<typename T, std::size_t Rows, std::size_t Cols>
+mm::matrix<T, Rows, Cols> operator*(
+ const T& k,
+ mm::matrix<T, Rows, Cols> a
+) {
+ return a *= k;
}
-/* Square interface implementation */
+// simple multiplication
+template<typename T, std::size_t M, std::size_t P1, std::size_t P2, std::size_t N>
+mm::matrix<T, M, N> operator*(
+ const mm::matrix<T, M, P1>& a,
+ const mm::matrix<T, P2, N>& b
+) {
+ static_assert(P1 == P2, "invalid matrix multiplication");
+ assert(a.cols() == b.rows());
-template<typename T, std::size_t N>
-T mm::square_interface<T, N>::trace()
-{
- T sum = 0;
- for (mm::diag_iterator<T, N> it(*this, 0); it.ok(); ++it)
- sum += *it;
+ mm::matrix<T, M, N> result;
+ mm::matrix<T, P2, N> bt = b.t(); // weak transposition
- return sum;
+ for (unsigned row = 0; row < M; row++)
+ for (unsigned col = 0; col < N; col++)
+ result.at(row, col) = a[row] * bt[col]; // scalar product
+
+ return result;
}
+// transposed multiplication
+/*template<typename T, std::size_t M, std::size_t P1, std::size_t P2, std::size_t N>
+mm::matrix<T, M, N> operator*(
+ const mm::matrix<T, P1, M>& a,
+ const mm::matrix<T, P2, N>& b
+) {
+ static_assert(P1 == P2, "invalid matrix multiplication");
+ assert(a.cols() == b.rows());
+
+ mm::matrix<T, M, N> result;
+ mm::matrix<T, P2, N> bt = b.t(); // weak transposition
+
+ for (unsigned row = 0; row < M; row++)
+ for (unsigned col = 0; col < N; col++)
+ result.at(row, col) = a[row] * bt[col]; // scalar product
+
+ return result;
+}*/
+
+
+/*template<typename T, std::size_t N>
+void mm::square_matrix<T, N>::transpose() {
+ for (unsigned row = 0; row < N; row++)
+ for (unsigned col = 0; col < row; col++)
+ std::swap(this->at(row, col), this->at(col, row));
+}*/
+
+
+/*
+ * Matrix operator <<
+ */
+
+template<typename T, std::size_t Rows, std::size_t Cols, unsigned NumW = 3>
+std::ostream& operator<<(std::ostream& os, const mm::matrix<T, Rows, Cols>& m) {
+
+ for (unsigned index = 0; index < m.rows(); index++) {
+ os << "[ ";
+ for (unsigned col = 0; col < m.cols()-1; ++col) {
+ os << std::setw(NumW) << m.at(index, col) << ", ";
+ }
+ os << std::setw(NumW) << m.at(index, m.cols()-1) << " ]\n";
+ }
+
+ return os;
+}