matrix utility
This commit is contained in:
commit
fc1e5a7e8d
50
.gitignore
vendored
Normal file
50
.gitignore
vendored
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
/.idea/
|
||||||
|
|
||||||
|
|
||||||
|
# common
|
||||||
|
*.csv
|
||||||
|
*.obj
|
||||||
|
*.exe
|
||||||
|
*.lst
|
||||||
|
*.tmp
|
||||||
|
*.pdb
|
||||||
|
|
||||||
|
# tex
|
||||||
|
*.aux
|
||||||
|
*.lof
|
||||||
|
*.log
|
||||||
|
*.lot
|
||||||
|
*.fls
|
||||||
|
*.out
|
||||||
|
*.toc
|
||||||
|
*.fmt
|
||||||
|
*.fot
|
||||||
|
*.cb
|
||||||
|
*.cb2
|
||||||
|
.*.lb
|
||||||
|
*.pdf
|
||||||
|
|
||||||
|
/cmake-*/
|
||||||
|
|
||||||
|
/sprawozdanie/*-figure*
|
||||||
|
|
||||||
|
## Build tool auxiliary files:
|
||||||
|
*.fdb_latexmk
|
||||||
|
*.synctex
|
||||||
|
*.synctex(busy)
|
||||||
|
*.synctex.gz
|
||||||
|
*.synctex.gz(busy)
|
||||||
|
*.pdfsync
|
||||||
|
*.auxlock
|
||||||
|
|
||||||
|
# hyperref
|
||||||
|
*.brf
|
||||||
|
|
||||||
|
# listings
|
||||||
|
*.lol
|
||||||
|
|
||||||
|
# makeidx
|
||||||
|
*.idx
|
||||||
|
*.ilg
|
||||||
|
*.ind
|
||||||
|
*.ist
|
7
CMakeLists.txt
Normal file
7
CMakeLists.txt
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
cmake_minimum_required(VERSION 3.10)
|
||||||
|
project(P02)
|
||||||
|
|
||||||
|
set(CMAKE_CXX_STANDARD 11)
|
||||||
|
set(CMAKE_C_STANDARD 11)
|
||||||
|
|
||||||
|
add_executable(main main.cpp)
|
29
main.cpp
Normal file
29
main.cpp
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
#include "matrix.h"
|
||||||
|
#include <iostream>
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
const size_t N = 10;
|
||||||
|
|
||||||
|
std::pair<Matrix<double>, Matrix<double>> prepare(size_t index, size_t n) {
|
||||||
|
double a1, a2, a3;
|
||||||
|
a2 = a3 = -1;
|
||||||
|
a1 = 5 + (index / 100) % 10;
|
||||||
|
|
||||||
|
auto M = Matrix<double>::diag(n, a1)
|
||||||
|
+ Matrix<double>::diag(n, a2, 1) + Matrix<double>::diag(n, a2, -1)
|
||||||
|
+ Matrix<double>::diag(n, a3, 2) + Matrix<double>::diag(n, a3, -2);
|
||||||
|
|
||||||
|
Matrix<double> b(n, 1);
|
||||||
|
for (size_t i = 0; i < n; ++i) {
|
||||||
|
b(i, 0) = sin((i + 1)*((index / 1000) % 10 + 1));
|
||||||
|
}
|
||||||
|
|
||||||
|
return std::make_pair(M, b);
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, const char argv[])
|
||||||
|
{
|
||||||
|
auto tuple = prepare(165581, N);
|
||||||
|
auto M = std::get<0>(tuple);
|
||||||
|
auto b = std::get<1>(tuple);
|
||||||
|
}
|
233
matrix.h
Normal file
233
matrix.h
Normal file
@ -0,0 +1,233 @@
|
|||||||
|
#ifndef P02_MATRIX_H
|
||||||
|
#define P02_MATRIX_H
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
#include <vector>
|
||||||
|
#include <cassert>
|
||||||
|
#include <ostream>
|
||||||
|
#include <iostream>
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
#define NORM_INF INT_MAX
|
||||||
|
|
||||||
|
using std::size_t;
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
class Matrix {
|
||||||
|
using self_t = Matrix<T>;
|
||||||
|
|
||||||
|
std::unique_ptr<T[]> values;
|
||||||
|
size_t _rows, _cols;
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit Matrix(size_t n);
|
||||||
|
Matrix(std::initializer_list<std::initializer_list<T>> values);
|
||||||
|
Matrix(size_t n, size_t m);
|
||||||
|
|
||||||
|
Matrix(const self_t& m);
|
||||||
|
|
||||||
|
T& operator()(size_t n, size_t m);
|
||||||
|
T operator()(size_t n, size_t m) const;
|
||||||
|
|
||||||
|
self_t operator*(const self_t& rhs);
|
||||||
|
self_t operator+(const self_t& rhs);
|
||||||
|
|
||||||
|
self_t operator*(T rhs);
|
||||||
|
self_t operator+(T rhs);
|
||||||
|
|
||||||
|
self_t& operator=(self_t&& rhs) noexcept;
|
||||||
|
|
||||||
|
size_t rows() const;
|
||||||
|
size_t cols() const;
|
||||||
|
|
||||||
|
template <typename U>
|
||||||
|
friend std::ostream &operator<<(std::ostream &os, const Matrix<U> &matrix);
|
||||||
|
|
||||||
|
static self_t diag(size_t n, T value, int offset = 0);
|
||||||
|
static self_t hvec(std::initializer_list<T> values);
|
||||||
|
static self_t vvec(std::initializer_list<T> values);
|
||||||
|
// static self_t diag(std::initializer_list values, size_t offset = 0);
|
||||||
|
};
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T>::Matrix(size_t n, size_t m) : _rows(n), _cols(m), values(new T[m*n]) {
|
||||||
|
for(size_t i = 0; i < m*n; i++) {
|
||||||
|
this->values.get()[i] = T();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T>::Matrix(size_t n) : Matrix(n, n) { }
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T>::Matrix(const Matrix<T> &m) : Matrix(m._rows, m._cols) {
|
||||||
|
std::copy(m.values.get(), m.values.get() + m._cols*m._rows, values.get());
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
T &Matrix<T>::operator()(size_t n, size_t m) {
|
||||||
|
return values.get()[n*this->_cols + m];
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
T Matrix<T>::operator()(size_t n, size_t m) const {
|
||||||
|
return values.get()[n*this->_cols + m];
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T> Matrix<T>::operator+(const Matrix<T> &rhs) {
|
||||||
|
assert(this->_cols == rhs._cols && this->_rows == rhs._rows);
|
||||||
|
self_t result(this->_rows, this->_cols);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < _rows; i++)
|
||||||
|
for (size_t j = 0; j < _cols; j++)
|
||||||
|
result(i, j) = (*this)(i, j) + rhs(i, j);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T> Matrix<T>::operator*(const Matrix<T> &rhs) {
|
||||||
|
assert(_cols == rhs._rows);
|
||||||
|
self_t result(_rows, rhs._cols);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < _rows; i++) {
|
||||||
|
for (size_t j = 0; j < rhs._cols; j++) {
|
||||||
|
T accumulator = 0;
|
||||||
|
for(size_t k = 0; k < _cols; k++)
|
||||||
|
accumulator += (*this)(i, k) * rhs(k, j);
|
||||||
|
|
||||||
|
result(i, j) = std::move(accumulator);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T> Matrix<T>::operator*(const T rhs) {
|
||||||
|
self_t result(*this);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < _rows; i++)
|
||||||
|
for (size_t j = 0; j < _cols; j++)
|
||||||
|
result(i, j) *= rhs;
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T> Matrix<T>::operator+(const T rhs) {
|
||||||
|
self_t result(*this);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < _rows; i++)
|
||||||
|
for (size_t j = 0; j < _cols; j++)
|
||||||
|
result(i, j) += rhs;
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
std::ostream &operator<<(std::ostream &os, const Matrix<T> &matrix) {
|
||||||
|
os << "[ ";
|
||||||
|
|
||||||
|
for (size_t i = 0; i < matrix._rows; i++) {
|
||||||
|
for (size_t j = 0; j < matrix._cols; j++) {
|
||||||
|
os << matrix(i, j) << " ";
|
||||||
|
}
|
||||||
|
os << (i == matrix._rows - 1 ? "]\n" : "\n ");
|
||||||
|
}
|
||||||
|
|
||||||
|
return os;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T> Matrix<T>::diag(size_t n, T value, int offset) {
|
||||||
|
self_t result(n);
|
||||||
|
if (offset >= 0)
|
||||||
|
for(size_t i = 0; i < n - offset; i++)
|
||||||
|
result(i,i+offset) = value;
|
||||||
|
else
|
||||||
|
for(size_t i = 0; i < n - offset; i++)
|
||||||
|
result(i-offset, i) = value;
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T>::Matrix(std::initializer_list<std::initializer_list<T>> values) : Matrix(values.size(), values.begin()->size()) {
|
||||||
|
size_t i = 0;
|
||||||
|
for (auto h : values) {
|
||||||
|
if (i > _rows) break;
|
||||||
|
|
||||||
|
size_t j = 0;
|
||||||
|
for (auto x : h) {
|
||||||
|
if (j > _cols) break;
|
||||||
|
|
||||||
|
(*this)(i, j) = x;
|
||||||
|
j++;
|
||||||
|
}
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T>& Matrix<T>::operator=(Matrix<T> &&rhs) noexcept {
|
||||||
|
this->_cols = rhs._cols;
|
||||||
|
this->_rows = rhs._rows;
|
||||||
|
std::swap(values, rhs.values);
|
||||||
|
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
inline size_t Matrix<T>::rows() const {
|
||||||
|
return this->_rows;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
inline size_t Matrix<T>::cols() const {
|
||||||
|
return this->_cols;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T> Matrix<T>::hvec(std::initializer_list<T> values) {
|
||||||
|
Matrix<T> result(1, values.size());
|
||||||
|
|
||||||
|
size_t i = 0;
|
||||||
|
for(auto val : values) {
|
||||||
|
result(0, i++) = val;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
Matrix<T> Matrix<T>::vvec(std::initializer_list<T> values) {
|
||||||
|
Matrix<T> result(values.size(), 1);
|
||||||
|
|
||||||
|
size_t i = 0;
|
||||||
|
for(auto val : values) {
|
||||||
|
result(i++, 0) = val;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
double norm(const Matrix<T>& matrix, unsigned norm = 2)
|
||||||
|
{
|
||||||
|
// needed?
|
||||||
|
// assert(matrix.cols() == 1 || matrix.rows() == 1);
|
||||||
|
|
||||||
|
double accumulator = 0.;
|
||||||
|
|
||||||
|
for (size_t i = 0; i < matrix.cols(); ++i) {
|
||||||
|
for (size_t j = 0; j < matrix.rows(); ++j) {
|
||||||
|
accumulator += std::pow(matrix(j, i), norm);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return std::pow(accumulator, 1. / norm);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif //P02_MATRIX_H
|
Loading…
Reference in New Issue
Block a user