namespace std::linalg {
template<in-matrix InMat1,
in-matrix InMat2,
possibly-packed-inout-matrix InOutMat,
class Triangle>
void symmetric_matrix_rank_2k_update(
InMat1 A,
InMat2 B,
InOutMat C,
Triangle t); // (1)
template<class ExecutionPolicy,
in-matrix InMat1,
in-matrix InMat2,
possibly-packed-inout-matrix InOutMat,
class Triangle>
void symmetric_matrix_rank_2k_update(
ExecutionPolicy&& exec,
InMat1 A,
InMat2 B,
InOutMat C,
Triangle t); // (2)
}
概要
対称かつ共役を取らないrank-2k updateを対称行列に行う。
引数tは対称行列の成分が上三角にあるのか、それとも下三角にあるのかを示す。
- (1): $C \leftarrow C + AB^T + BA^T$
- (2): (1)を指定された実行ポリシーで実行する。
適格要件
- 共通:
Triangleはupper_triangle_tまたはlower_triangle_tInMatがlayout_blas_packedを持つなら、レイアウトのTriangleテンプレート引数とこの関数のTriangleテンプレート引数が同じ型possibly-addable<decltype(A), decltype(B), decltype(C)>()がtruecompatible-static-extents<decltype(A), decltype(A)>(0, 1)がtrue(つまりAが正方行列であること)
- (2):
is_execution_policy<ExecutionPolicy>::valueがtrue
事前条件
A.extent(0) == A.extent(1)addable(A, B, C)がtrue
効果
$C \leftarrow C + AB^T + BA^T$
戻り値
なし
計算量
$O(\verb|A.extent(0)| \times \verb|A.extent(1)| \times \verb|C.extent(0)|)$
例
[注意] 処理系にあるコンパイラで確認していないため、間違っているかもしれません。
#include <array>
#include <iostream>
#include <linalg>
#include <mdspan>
#include <vector>
template <class Matrix>
void print_mat(const Matrix& A) {
for(int i = 0; i < A.extent(0); ++i) {
for(int j = 0; j < i; ++j) {
std::cout << A[j, i] << ' ';
}
for(int j = i; j < A.extent(1) - 1; ++j) {
std::cout << A[i, j] << ' ';
}
std::cout << A[i, A.extent(1) - 1] << '\n';
}
}
template <class Matrix>
void init_mat(Matrix& A, double geta = 0.0) {
for(int i = 0; i < A.extent(0); ++i) {
for(int j = 0; j < A.extent(1); ++j) {
A[i,j] = A.extent(1) * i + j + geta;
}
}
}
template <class Matrix>
void init_symm_mat(Matrix& A) {
for(int i = 0; i < A.extent(0); ++i) {
for(int j = i; j < A.extent(1); ++j) {
A[i, j] = i * A.extent(1) + j;
}
}
}
int main()
{
constexpr size_t N = 2;
std::vector<double> A_vec(N * N);
std::vector<double> B_vec(N * N);
std::vector<double> C_vec(N * N);
std::mdspan A(A_vec.data());
std::mdspan B(B_vec.data());
std::mdspan<
double,
std::extents<size_t, N, N>,
std::linalg::layout_blas_packed<
std::linalg::upper_triangle_t,
std::linalg::row_major_t>
> C(C_vec.data());
init_mat(A);
init_mat(B, 1);
init_symm_mat(C);
// (1)
std::cout << "(1)\n";
std::linalg::symmetric_matrix_rank_2k_update(
A,
B,
C,
std::linalg::upper_triangle);
print_mat(C);
// (2)
init_symm_mat(C);
std::cout << "(2)\n";
std::linalg::symmetric_matrix_rank_2k_update(
std::execution::par,
A,
B,
C,
std::linalg::upper_triangle);
print_mat(C);
return 0;
}
出力
(1)
4 13
13 39
(2)
4 13
13 39
バージョン
言語
- C++26
処理系
- Clang: ??
- GCC: ??
- ICC: ??
- Visual C++: ??