|
|
@ -7,99 +7,100 @@ |
|
|
|
|
|
|
|
|
|
|
|
namespace amt { |
|
|
|
namespace amt { |
|
|
|
|
|
|
|
|
|
|
|
template <typename T> |
|
|
|
template <typename T> |
|
|
|
requires std::is_integral_v<T> |
|
|
|
requires std::is_integral_v<T> |
|
|
|
struct Matrix { |
|
|
|
struct Matrix { |
|
|
|
using base_type = std::vector<T>; |
|
|
|
using base_type = std::vector<T>; |
|
|
|
using value_type = typename base_type::value_type; |
|
|
|
using value_type = typename base_type::value_type; |
|
|
|
using pointer = typename base_type::pointer; |
|
|
|
using pointer = typename base_type::pointer; |
|
|
|
using const_pointer = typename base_type::const_pointer; |
|
|
|
using const_pointer = typename base_type::const_pointer; |
|
|
|
using reference = typename base_type::reference; |
|
|
|
using reference = typename base_type::reference; |
|
|
|
using const_reference = typename base_type::const_reference; |
|
|
|
using const_reference = typename base_type::const_reference; |
|
|
|
using iterator = typename base_type::iterator; |
|
|
|
using iterator = typename base_type::iterator; |
|
|
|
using const_iterator = typename base_type::const_iterator; |
|
|
|
using const_iterator = typename base_type::const_iterator; |
|
|
|
using size_type = std::size_t; |
|
|
|
using size_type = std::size_t; |
|
|
|
|
|
|
|
|
|
|
|
template <bool IsConst> |
|
|
|
template <bool IsConst> |
|
|
|
struct View { |
|
|
|
struct View { |
|
|
|
using base_type = std::conditional_t<IsConst, const_pointer, pointer>; |
|
|
|
using base_type = std::conditional_t<IsConst, const_pointer, pointer>; |
|
|
|
base_type data; |
|
|
|
base_type data; |
|
|
|
size_type size; |
|
|
|
size_type size; |
|
|
|
|
|
|
|
|
|
|
|
constexpr auto operator[](size_type k) const noexcept { |
|
|
|
constexpr reference operator[](size_type k) noexcept requires (!IsConst) { |
|
|
|
assert(k < size && "Out of bound access"); |
|
|
|
assert(k < size && "Out of bound access"); |
|
|
|
return data[k]; |
|
|
|
return data[k]; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
constexpr auto operator[](size_type k) noexcept requires (!IsConst) { |
|
|
|
constexpr const_reference operator[](size_type k) const noexcept { |
|
|
|
assert(k < size && "Out of bound access"); |
|
|
|
assert(k < size && "Out of bound access"); |
|
|
|
return data[k]; |
|
|
|
return data[k]; |
|
|
|
} |
|
|
|
} |
|
|
|
}; |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
constexpr Matrix() noexcept = default; |
|
|
|
|
|
|
|
constexpr Matrix(Matrix const&) = default; |
|
|
|
constexpr Matrix() noexcept = default; |
|
|
|
constexpr Matrix& operator=(Matrix const&) = default; |
|
|
|
constexpr Matrix(Matrix const&) = default; |
|
|
|
constexpr Matrix(Matrix &&) noexcept = default; |
|
|
|
constexpr Matrix& operator=(Matrix const&) = default; |
|
|
|
constexpr Matrix& operator=(Matrix &&) noexcept = default; |
|
|
|
constexpr Matrix(Matrix &&) noexcept = default; |
|
|
|
constexpr ~Matrix() = default; |
|
|
|
constexpr Matrix& operator=(Matrix &&) noexcept = default; |
|
|
|
|
|
|
|
constexpr ~Matrix() = default; |
|
|
|
|
|
|
|
|
|
|
|
Matrix(size_type row, size_type col, value_type def = {}) |
|
|
|
|
|
|
|
: m_data(row * col, def) |
|
|
|
Matrix(size_type row, size_type col, value_type def = {}) |
|
|
|
, m_row(row) |
|
|
|
: m_data(row * col, def) |
|
|
|
, m_col(col) |
|
|
|
, m_row(row) |
|
|
|
{} |
|
|
|
, m_col(col) |
|
|
|
|
|
|
|
{} |
|
|
|
constexpr bool empty() const noexcept { return m_data.empty(); }; |
|
|
|
|
|
|
|
constexpr bool size() const noexcept { return m_data.size(); }; |
|
|
|
constexpr bool empty() const noexcept { return m_data.empty(); }; |
|
|
|
constexpr size_type rows() const noexcept { return m_row; }; |
|
|
|
constexpr bool size() const noexcept { return m_data.size(); }; |
|
|
|
constexpr size_type cols() const noexcept { return m_col; }; |
|
|
|
constexpr size_type rows() const noexcept { return m_row; }; |
|
|
|
|
|
|
|
constexpr size_type cols() const noexcept { return m_col; }; |
|
|
|
constexpr iterator begin() noexcept { return m_data.begin(); } |
|
|
|
|
|
|
|
constexpr iterator end() noexcept { return m_data.end(); } |
|
|
|
constexpr iterator begin() noexcept { return m_data.begin(); } |
|
|
|
constexpr const_iterator begin() const noexcept { return m_data.begin(); } |
|
|
|
constexpr iterator end() noexcept { return m_data.end(); } |
|
|
|
constexpr const_iterator end() const noexcept { return m_data.end(); } |
|
|
|
constexpr const_iterator begin() const noexcept { return m_data.begin(); } |
|
|
|
|
|
|
|
constexpr const_iterator end() const noexcept { return m_data.end(); } |
|
|
|
constexpr auto operator()(size_type r, size_type c) noexcept -> reference { |
|
|
|
|
|
|
|
auto const index = r * m_col + c; // column-major;
|
|
|
|
constexpr auto operator()(size_type r, size_type c) noexcept -> reference { |
|
|
|
assert(index < size() && "Out of bound access"); |
|
|
|
auto const index = r * m_col + c; // column-major;
|
|
|
|
return m_data[index]; |
|
|
|
assert(index < size() && "Out of bound access"); |
|
|
|
} |
|
|
|
return m_data[index]; |
|
|
|
|
|
|
|
} |
|
|
|
constexpr auto operator()(size_type r, size_type c) const noexcept -> const_reference { |
|
|
|
|
|
|
|
auto const index = r * m_col + c; // column-major;
|
|
|
|
constexpr auto operator()(size_type r, size_type c) const noexcept -> const_reference { |
|
|
|
assert(index < size() && "Out of bound access"); |
|
|
|
auto const index = r * m_col + c; // column-major;
|
|
|
|
return m_data[index]; |
|
|
|
assert(index < size() && "Out of bound access"); |
|
|
|
} |
|
|
|
return m_data[index]; |
|
|
|
|
|
|
|
} |
|
|
|
constexpr auto operator[](size_type r) noexcept -> View<false> { |
|
|
|
|
|
|
|
auto const base = r * m_col; |
|
|
|
constexpr auto operator[](size_type r) noexcept -> View<false> { |
|
|
|
assert(r < rows() && "Out of bound access"); |
|
|
|
auto const base = r * m_col; |
|
|
|
return { .data = m_data.data() + base, .size = m_col }; |
|
|
|
assert(r < rows() && "Out of bound access"); |
|
|
|
} |
|
|
|
return { .data = m_data.data() + base, .size = m_col }; |
|
|
|
|
|
|
|
} |
|
|
|
constexpr auto operator[](size_type r) const noexcept -> View<true> { |
|
|
|
|
|
|
|
auto const base = r * m_col; |
|
|
|
constexpr auto operator[](size_type r) const noexcept -> View<true> { |
|
|
|
assert(r < rows() && "Out of bound access"); |
|
|
|
auto const base = r * m_col; |
|
|
|
return { .data = m_data.data() + base, .size = m_col }; |
|
|
|
assert(r < rows() && "Out of bound access"); |
|
|
|
} |
|
|
|
return { .data = m_data.data() + base, .size = m_col }; |
|
|
|
|
|
|
|
} |
|
|
|
friend std::ostream& operator<<(std::ostream& os, Matrix const& m) { |
|
|
|
|
|
|
|
os << "[\n"; |
|
|
|
friend std::ostream& operator<<(std::ostream& os, Matrix const& m) { |
|
|
|
for (auto i = size_type{}; i < m.rows(); ++i) { |
|
|
|
os << "[\n"; |
|
|
|
for (auto j = size_type{}; j < m.cols(); ++j) { |
|
|
|
for (auto i = size_type{}; i < m.rows(); ++i) { |
|
|
|
os << m[i][j] << ", "; |
|
|
|
for (auto j = size_type{}; j < m.cols(); ++j) { |
|
|
|
} |
|
|
|
os << m[i][j] << ", "; |
|
|
|
os << '\n'; |
|
|
|
} |
|
|
|
} |
|
|
|
os << '\n'; |
|
|
|
return os << "]"; |
|
|
|
} |
|
|
|
} |
|
|
|
return os << "]"; |
|
|
|
|
|
|
|
} |
|
|
|
private: |
|
|
|
|
|
|
|
base_type m_data; |
|
|
|
private: |
|
|
|
size_type m_row{}; |
|
|
|
base_type m_data; |
|
|
|
size_type m_col{}; |
|
|
|
size_type m_row{}; |
|
|
|
}; |
|
|
|
size_type m_col{}; |
|
|
|
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
} // namespace amt
|
|
|
|
} // namespace amt
|
|
|
|