Line data Source code
1 : #include "Matrix.hpp"
2 :
3 : #include <cmath>
4 : #include <iomanip>
5 : #include <sstream>
6 : #include <stdexcept>
7 :
8 : #include <iostream>
9 :
10 : namespace HLR
11 : {
12 : namespace Kinematics
13 : {
14 : template<typename T, std::size_t m, std::size_t n>
15 : constexpr Matrix<T, m, n> Matrix<T, m, n>::get_identity() noexcept
16 : {
17 : auto identity = Matrix<T, m, m>();
18 :
19 : for (std::size_t i = 0; i < m; ++i)
20 : {
21 : identity[i][i] = 1;
22 : }
23 :
24 : return identity;
25 : }
26 :
27 : template<typename T, std::size_t m, std::size_t n>
28 3 : constexpr Matrix<T, m, n>::Matrix(
29 3 : const std::initializer_list<std::initializer_list<T>> matrix_data)
30 : {
31 3 : if (matrix_data.size() != m)
32 : {
33 1 : throw std::invalid_argument("Invalid amount of rows.");
34 : }
35 :
36 3 : for (const auto& col : matrix_data)
37 : {
38 2 : if (col.size() != n)
39 : {
40 1 : throw std::invalid_argument("Invalid amount of columns.");
41 : }
42 : }
43 :
44 1 : std::size_t pos_i = 0;
45 1 : std::size_t pos_j = 0;
46 :
47 2 : for (auto i = matrix_data.begin(); i != matrix_data.end(); ++i)
48 : {
49 2 : for (auto j = i->begin(); j != i->end(); ++j)
50 : {
51 1 : this->data[pos_i][pos_j] = *j;
52 1 : ++pos_j;
53 : }
54 1 : ++pos_i;
55 1 : pos_j = 0;
56 : }
57 1 : }
58 :
59 : template<typename T, std::size_t m, std::size_t n>
60 2434101 : constexpr std::size_t Matrix<T, m, n>::get_m() const noexcept
61 : {
62 2434101 : return m;
63 : }
64 :
65 : template<typename T, std::size_t m, std::size_t n>
66 1 : constexpr std::size_t Matrix<T, m, n>::get_n() const noexcept
67 : {
68 1 : return n;
69 : }
70 :
71 : template<typename T, std::size_t m, std::size_t n>
72 70476706 : constexpr std::array<T, n>& Matrix<T, m, n>::operator[](
73 : const std::size_t pos) noexcept
74 : {
75 70476706 : return this->data[pos];
76 : }
77 :
78 : template<typename T, std::size_t m, std::size_t n>
79 45120924 : constexpr const std::array<T, n>& Matrix<T, m, n>::operator[](
80 : const std::size_t pos) const noexcept
81 : {
82 45120924 : return this->data[pos];
83 : }
84 :
85 : template<typename T, std::size_t m, std::size_t n>
86 2 : constexpr std::array<T, n>& Matrix<T, m, n>::at(const std::size_t pos)
87 : {
88 2 : if (pos >= m)
89 : {
90 1 : throw std::out_of_range("Index out of range.");
91 : }
92 :
93 1 : return this->data.at(pos);
94 : }
95 :
96 : template<typename T, std::size_t m, std::size_t n>
97 2 : constexpr const std::array<T, n>& Matrix<T, m, n>::at(
98 : const std::size_t pos) const
99 : {
100 2 : if (pos >= m)
101 : {
102 1 : throw std::out_of_range("Index out of range.");
103 : }
104 :
105 1 : return this->data.at(pos);
106 : }
107 :
108 : template<typename T, std::size_t m, std::size_t n>
109 207139 : constexpr bool Matrix<T, m, n>::fuzzy_equal(const Matrix<T, m, n>& mat,
110 : T fuzz) const noexcept
111 : {
112 826272 : for (std::size_t i = 0; i < m; ++i)
113 : {
114 2465877 : for (std::size_t j = 0; j < n; ++j)
115 : {
116 1846744 : if (std::abs((*this)[i][j] - mat[i][j]) > fuzz)
117 : {
118 1385 : return false;
119 : }
120 : }
121 : }
122 :
123 205754 : return true;
124 : }
125 :
126 : template<typename T, std::size_t m, std::size_t n>
127 977 : constexpr bool Matrix<T, m, n>::operator==(const Matrix<T, m, n>& mat) const
128 : noexcept
129 : {
130 977 : return this->fuzzy_equal(mat, 0);
131 : }
132 :
133 : template<typename T, std::size_t m, std::size_t n>
134 3 : constexpr bool Matrix<T, m, n>::operator!=(const Matrix<T, m, n>& mat) const
135 : noexcept
136 : {
137 3 : return !(*this == mat);
138 : }
139 :
140 : template<typename T, std::size_t m, std::size_t n>
141 8 : constexpr void Matrix<T, m, n>::operator+=(T scalar) noexcept
142 : {
143 24 : for (std::size_t i = 0; i < m; ++i)
144 : {
145 64 : for (std::size_t j = 0; j < n; ++j)
146 : {
147 48 : (*this)[i][j] += scalar;
148 : }
149 : }
150 8 : }
151 :
152 : template<typename T, std::size_t m, std::size_t n>
153 4 : constexpr void Matrix<T, m, n>::operator-=(T scalar) noexcept
154 : {
155 4 : (*this) += -1 * scalar;
156 4 : }
157 :
158 : template<typename T, std::size_t m, std::size_t n>
159 1207528 : constexpr void Matrix<T, m, n>::operator*=(T scalar) noexcept
160 : {
161 5630904 : for (std::size_t i = 0; i < m; ++i)
162 : {
163 8846776 : for (std::size_t j = 0; j < n; ++j)
164 : {
165 4423400 : (*this)[i][j] *= scalar;
166 : }
167 : }
168 1207528 : }
169 :
170 : template<typename T, std::size_t m, std::size_t n>
171 4 : constexpr void Matrix<T, m, n>::operator/=(T scalar) noexcept
172 : {
173 12 : for (std::size_t i = 0; i < m; ++i)
174 : {
175 32 : for (std::size_t j = 0; j < n; ++j)
176 : {
177 24 : (*this)[i][j] /= scalar;
178 : }
179 : }
180 4 : }
181 :
182 : template<typename T, std::size_t m, std::size_t n>
183 1002279 : constexpr void Matrix<T, m, n>::operator+=(const Matrix<T, m, n>& mat) noexcept
184 : {
185 4421714 : for (std::size_t i = 0; i < m; ++i)
186 : {
187 6838886 : for (std::size_t j = 0; j < n; ++j)
188 : {
189 3419451 : (*this)[i][j] += mat[i][j];
190 : }
191 : }
192 1002279 : }
193 :
194 : template<typename T, std::size_t m, std::size_t n>
195 603825 : constexpr void Matrix<T, m, n>::operator-=(const Matrix<T, m, n>& mat) noexcept
196 : {
197 603825 : (*this) += mat * static_cast<T>(-1);
198 603825 : }
199 :
200 : template<typename T, std::size_t m, std::size_t n>
201 397512 : constexpr Matrix<T, m, n> Matrix<T, m, n>::operator+(
202 : const Matrix<T, m, n>& mat) const noexcept
203 : {
204 397512 : Matrix<T, m, n> copy(*this);
205 397512 : copy += mat;
206 397512 : return copy;
207 : }
208 :
209 : template<typename T, std::size_t m, std::size_t n>
210 603824 : constexpr Matrix<T, m, n> Matrix<T, m, n>::operator-(
211 : const Matrix<T, m, n>& mat) const noexcept
212 : {
213 603824 : Matrix<T, m, n> copy(*this);
214 603824 : copy -= mat;
215 603824 : return copy;
216 : }
217 :
218 : template<typename T, std::size_t m, std::size_t n>
219 : template<std::size_t p>
220 807244 : constexpr Matrix<T, m, p> Matrix<T, m, n>::operator*(
221 : const Matrix<T, n, p>& mat) const noexcept
222 : {
223 807244 : Matrix<T, m, p> product;
224 :
225 2431821 : for (std::size_t i = 0; i < m; ++i)
226 : {
227 4475180 : for (std::size_t j = 0; j < p; ++j)
228 : {
229 11402620 : for (std::size_t k = 0; k < n; ++k)
230 : {
231 8552017 : product[i][j] += (*this)[i][k] * mat[k][j];
232 : }
233 : }
234 : }
235 :
236 807244 : return product;
237 : }
238 :
239 : template<typename T, std::size_t m, std::size_t n>
240 : template<std::size_t p>
241 204337 : constexpr Matrix<T, m, n + p> Matrix<T, m, n>::horizontal_augment(
242 : const Matrix<T, m, p>& mat) const noexcept
243 : {
244 204337 : Matrix<T, m, n + p> augmented;
245 :
246 817348 : for (std::size_t i = 0; i < m; ++i)
247 : {
248 2452042 : for (std::size_t j = 0; j < n; ++j)
249 : {
250 1839031 : augmented[i][j] = (*this)[i][j];
251 : }
252 :
253 2452046 : for (std::size_t j = n; j < n + p; ++j)
254 : {
255 1839035 : augmented[i][j] = mat[i][j - n];
256 : }
257 : }
258 :
259 204337 : return augmented;
260 : }
261 :
262 : template<typename T, std::size_t m, std::size_t n>
263 : template<std::size_t p>
264 2 : constexpr Matrix<T, m + p, n> Matrix<T, m, n>::vertical_augment(
265 : const Matrix<T, p, n>& mat) const noexcept
266 : {
267 2 : Matrix<T, m + p, n> augmented;
268 :
269 5 : for (std::size_t i = 0; i < n; ++i)
270 : {
271 10 : for (std::size_t j = 0; j < m; ++j)
272 : {
273 7 : augmented[j][i] = (*this)[j][i];
274 : }
275 :
276 11 : for (std::size_t j = m; j < m + p; ++j)
277 : {
278 8 : augmented[j][i] = mat[j - m][i];
279 : }
280 : }
281 :
282 2 : return augmented;
283 : }
284 :
285 : template<typename T, std::size_t m, std::size_t n>
286 : template<std::size_t y, std::size_t x, std::size_t p, std::size_t q>
287 807129 : constexpr Matrix<T, p, q> Matrix<T, m, n>::slice() const noexcept
288 : {
289 807129 : Matrix<T, p, q> mat;
290 :
291 3228517 : for (std::size_t i = y; i < m && i - y < p; ++i)
292 : {
293 7294791 : for (std::size_t j = x; j < n && j - x < q; ++j)
294 : {
295 4873403 : mat[i - y][j - x] = (*this)[i][j];
296 : }
297 : }
298 :
299 807129 : return mat;
300 : }
301 :
302 : template<typename T, std::size_t m, std::size_t n>
303 398581 : constexpr Matrix<T, n, m> Matrix<T, m, n>::transpose() const noexcept
304 : {
305 398581 : Matrix<T, n, m> transposed;
306 :
307 1594496 : for (std::size_t i = 0; i < m; ++i)
308 : {
309 2391842 : for (std::size_t j = 0; j < n; ++j)
310 : {
311 1195927 : transposed[j][i] = (*this)[i][j];
312 : }
313 : }
314 :
315 398581 : return transposed;
316 : }
317 :
318 : template<typename T, std::size_t m, std::size_t n>
319 204335 : constexpr std::optional<Matrix<T, m, n>> Matrix<T, m, n>::inverse() const
320 : noexcept
321 : {
322 : static_assert(m == n,
323 : "Inverse matrix is only defined for square matrices.");
324 :
325 204335 : const auto identity = Matrix<T, m, n>::get_identity();
326 204335 : auto augmented = this->horizontal_augment(identity);
327 :
328 817338 : for (std::size_t i = 0; i < m; ++i)
329 : {
330 : /**
331 : * Find the diagonal element closest to -1 or 1.
332 : * And if it isn't in the current row swap it.
333 : * This is done to make the inverse mroe numerically stable.
334 : * Also check if that the row may not be zero since that would
335 : * result in a divide by zero error.
336 : */
337 613004 : std::size_t min_row_index = i;
338 613004 : std::optional<T> closest_to_one;
339 :
340 1839015 : for (std::size_t j = min_row_index; j < m; ++j)
341 : {
342 1226011 : if (std::abs(augmented[j][i]) <= std::numeric_limits<T>::min())
343 : {
344 254372 : continue;
345 : }
346 :
347 971639 : const T abs_value = std::abs(1 - std::abs(augmented[j][i]));
348 :
349 1943278 : if (!closest_to_one.has_value()
350 971639 : || abs_value < closest_to_one.value())
351 : {
352 888303 : closest_to_one = abs_value;
353 888303 : min_row_index = j;
354 : }
355 : }
356 :
357 613004 : if (!closest_to_one.has_value())
358 : {
359 1 : return std::nullopt;
360 : }
361 :
362 613003 : if (min_row_index != i)
363 : {
364 2277365 : for (std::size_t j = 0; j < 2 * m; ++j)
365 : {
366 1952028 : T tmp = augmented[i][j];
367 1952028 : augmented[i][j] = augmented[min_row_index][j];
368 1952028 : augmented[min_row_index][j] = tmp;
369 : }
370 : }
371 :
372 : // Divide the current row by a scalar so that the current diagonal
373 : // element will be 1.
374 613003 : const auto divisor = augmented[i][i];
375 :
376 4291029 : for (std::size_t j = 0; j < 2 * m; ++j)
377 : {
378 3678026 : augmented[i][j] /= divisor;
379 : }
380 :
381 : // Set all other rows to 0 by using the row that is 1.
382 2452016 : for (std::size_t j = 0; j < m; ++j)
383 : {
384 1839013 : if (i == j)
385 : {
386 613003 : continue;
387 : }
388 :
389 1226010 : const auto multiplier = augmented[j][i];
390 :
391 8582094 : for (std::size_t k = 0; k < 2 * m; ++k)
392 : {
393 7356084 : augmented[j][k] -= multiplier * augmented[i][k];
394 : }
395 : }
396 : }
397 :
398 : /**
399 : * Check if the inversed matrix times the original matrix is equal
400 : * to the identity matrix. The equal precision is the precision
401 : * when the identity matrix is considered equal.
402 : */
403 204334 : const auto inversed = augmented.template slice<0, m, m, m>();
404 204334 : const auto identity_check = inversed * (*this);
405 204334 : const auto equal_precision = 0.0000000001;
406 :
407 204334 : if (!identity_check.fuzzy_equal(identity, equal_precision))
408 : {
409 2 : return std::nullopt;
410 : }
411 :
412 204332 : return augmented.template slice<0, m, m, m>();
413 : }
414 :
415 : template<typename T, std::size_t m, std::size_t n>
416 398578 : constexpr T Matrix<T, m, n>::get_magnitude() const noexcept
417 : {
418 : static_assert(
419 : m == 1 || n == 1,
420 : "Magnitude of a matrix is only defined for column and row matrices.");
421 :
422 398578 : T magnitude = 0;
423 :
424 : if constexpr (m == 1) // NOLINT
425 : {
426 1 : magnitude =
427 2 : static_cast<T>(std::sqrt(((*this) * this->transpose())[0][0]));
428 : }
429 : else
430 : {
431 398577 : magnitude =
432 797154 : static_cast<T>(std::sqrt((this->transpose() * (*this))[0][0]));
433 : }
434 :
435 398578 : return magnitude;
436 : }
437 :
438 : template<typename T, std::size_t m, std::size_t n>
439 1 : std::string Matrix<T, m, n>::str() const
440 : {
441 2 : std::stringstream ss;
442 1 : ss << std::setprecision(9) << std::setw(6) << std::fixed;
443 :
444 3 : for (std::size_t i = 0; i < m; ++i)
445 : {
446 6 : for (std::size_t j = 0; j < n; ++j)
447 : {
448 4 : ss << (*this)[i][j] << "\t";
449 : }
450 2 : ss << std::endl;
451 : }
452 :
453 2 : return ss.str();
454 : }
455 :
456 : template<typename T, std::size_t m, std::size_t n>
457 2 : constexpr Matrix<T, m, n> operator+(const Matrix<T, m, n>& lhs, T rhs) noexcept
458 : {
459 2 : Matrix<T, m, n> copy(lhs);
460 2 : copy += rhs;
461 2 : return copy;
462 : }
463 :
464 : template<typename T, std::size_t m, std::size_t n>
465 1 : constexpr Matrix<T, m, n> operator+(T lhs, const Matrix<T, m, n>& rhs) noexcept
466 : {
467 1 : Matrix<T, m, n> copy(rhs);
468 1 : copy += lhs;
469 1 : return copy;
470 : }
471 :
472 : template<typename T, std::size_t m, std::size_t n>
473 2 : constexpr Matrix<T, m, n> operator-(const Matrix<T, m, n>& lhs, T rhs) noexcept
474 : {
475 2 : Matrix<T, m, n> copy(lhs);
476 2 : copy -= rhs;
477 2 : return copy;
478 : }
479 :
480 : template<typename T, std::size_t m, std::size_t n>
481 1 : constexpr Matrix<T, m, n> operator-(T lhs, const Matrix<T, m, n>& rhs) noexcept
482 : {
483 1 : Matrix<T, m, n> copy(rhs);
484 1 : copy -= lhs;
485 1 : return copy;
486 : }
487 :
488 : template<typename T, std::size_t m, std::size_t n>
489 808158 : constexpr Matrix<T, m, n> operator*(const Matrix<T, m, n>& lhs, T rhs) noexcept
490 : {
491 808158 : Matrix<T, m, n> copy(lhs);
492 808158 : copy *= rhs;
493 808158 : return copy;
494 : }
495 :
496 : template<typename T, std::size_t m, std::size_t n>
497 204331 : constexpr Matrix<T, m, n> operator*(T lhs, const Matrix<T, m, n>& rhs) noexcept
498 : {
499 204331 : Matrix<T, m, n> copy(rhs);
500 204331 : copy *= lhs;
501 204331 : return copy;
502 : }
503 :
504 : template<typename T, std::size_t m, std::size_t n>
505 2 : constexpr Matrix<T, m, n> operator/(const Matrix<T, m, n>& lhs, T rhs) noexcept
506 : {
507 2 : Matrix<T, m, n> copy(lhs);
508 2 : copy /= rhs;
509 2 : return copy;
510 : }
511 :
512 : template<typename T, std::size_t m, std::size_t n>
513 1 : constexpr Matrix<T, m, n> operator/(T lhs, const Matrix<T, m, n>& rhs) noexcept
514 : {
515 1 : Matrix<T, m, n> copy(rhs);
516 1 : copy /= lhs;
517 1 : return copy;
518 : }
519 :
520 : template<typename T, std::size_t m, std::size_t n>
521 1 : std::ostream& operator<<(std::ostream& os, const Matrix<T, m, n>& mat)
522 : {
523 1 : return os << mat.str();
524 : }
525 :
526 : } // namespace Kinematics
527 : } // namespace HLR
|