gpt4 book ai didi

c++ - 如何在 C++ 中计算列随机矩阵的特征向量

转载 作者:太空宇宙 更新时间:2023-11-04 12:22:18 25 4
gpt4 key购买 nike

我有一个列随机矩阵 A,想用 C++ 求解以下方程:斧=x

我假设我需要找出特征值设置为 1 的特征向量 x(对吗?),但我无法在 C++ 中计算出来。到目前为止,我已经检查了一些数学库,例如 Seldon、CPPScaLapack、Eigen...其中,Eigen 似乎是一个不错的选择,但我不明白如何利用它们中的任何一个来求解上面的方程。

你能给我一些解决方程式的建议/代码片段或想法吗?非常感谢任何帮助。

谢谢。

编辑:A 是 n×n 实数非负矩阵。

最佳答案

请记住,我没有使用 Eigen,但它是 EigenSolverSelfAdjointEigenSolver看起来能够解决这个问题。这些被列为“实验性”,因此可能存在错误,API 将来可能会更改。

// simple, not very efficient
template <typename _M>
bool isSelfAdjoint(const _M& a) {
return a == a.adjoint();
}

template <typename _M>
std::pair<Eigen::EigenSolver<_M>::EigenvalueType Eigen::EigenSolver<_M>::EigenvectorType>
eigenvectors(const _M& a) {
if (isSelfAdjoint(a)) {
Eigen::EigenSolver<M> saes(a);
return pair(saes.eigenvalues(), saes.eigenvectors());
} else {
Eigen::EigenSolver<M> es(a);
return pair(es.eigenvalues, es.eigenvectors());
}
}

这两个求解器类具有不同的特征值和特征向量集合类型,但由于它们都基于类 Matrix 并且矩阵是可转换的,所以上面的应该可以工作。

或者,您可以将问题视为 homogeneous linear equation (A-In) x = 0,将A-In转化为上三角矩阵即可求解。 Gaussian elimination会这样做(尽管您需要跳过确保前导系数为 1 的每一行的归一化步骤,因为整数不是字段)。快速浏览上述项目并没有发现对行梯队转换的支持,这可能意味着我错过了它。在任何情况下,使用一些辅助类(RowMajorRowMajor::iteratorRowWithPivot 在下文中)实现都不是太难。我什至没有测试过这是否会编译,所以将它更多地视为算法的说明,而不是完整的嵌入式解决方案。尽管该示例使用函数,但使用类 (à la EigenSolver) 可能更有意义。

/* Finds a row with the lowest pivot index in a range of matrix rows.
* Arguments:
* - start: the first row to check
* - end: row that ends search range (not included in search)
* - pivot_i (optional): if a row with pivot index == pivot_i is found, search
* no more. Can speed things up if the pivot index of all rows in the range
* have a known lower bound.
*
* Returns an iterator p where p->pivot_i = min([start .. end-1]->pivot_i)
*
*/
template <typename _M>
RowMajor<_M>::iterator
find_lead_pivot (RowMajor<_M>::iterator start,
const RowMajor<_M>::iterator& end,
int pivot_i=0)
{
RowMajor<_M>::iterator lead=start;
for (; start != end; ++start) {
if (start->pivot() <= pivot_i) {
return start;
}
if (start->pivot() < lead->pivot()) {
lead = start;
}
}
return end;
}

/* Returns a matrix that's the row echelon form of the passed in matrix.
*/
template <typename _M>
_M form_of_echelon(const _M& a) {
_M a_1 = a-_M::Identity();
RowMajor<_M> rma_1 = RowMajor<_M>(a_1);
typedef RowMajor<_M>::iterator RMIter;
RMIter lead;
int i=0;

/*
Loop invariant: row(i).pivot_i <= row(j).pivot_i, for j st. j>i
*/
for (RMIter row_i = rma_1.begin();
row_i != rma_1.end() && row_i->pivot() != 0;
++row_i, ++i)
{
lead = find_lead_pivot(row_i, rma_1.end(), i);
// ensure row(i) has minimal pivot index
swap(*lead, *row_i);

// ensure row(j).pivot_i > row(i).pivot_i
for (RMIter row_j = row_i+1;
row_j != rma_1.end();
++row_j)
{
*row_j = *row_j * row_i->pivot() - *row_i * row_j->pivot();
}
/* the best we can do towards true row echelon form is reduce
* the leading coefficient by the row's GCD
*/
// *row_i /= gcd(*row_i);
}
return static_cast<_M>(rma_1);
}

/* Converts a matrix to echelon form in-place
*/
template <typename _M>
_M& form_of_echelon(_M& a) {
a -= _M::Identity();
RowMajor<_M> rma_1 = RowMajor<_M>(a);
typedef RowMajor<_M>::iterator RMIter;
RMIter lead;
int i=0;

/*
Loop invariant: row(i).pivot_i <= row(j).pivot_i, for j st. j>i
*/
for (RMIter row_i = rma_1.begin();
row_i != rma_1.end() && row_i->pivot() != 0;
++row_i, ++i)
{
lead = find_lead_pivot(row_i, rma_1.end(), i);
// ensure row(i) has minimal pivot index
swap(*lead, *row_i);

for (RMIter row_j = row_i+1;
row_j != rma_1.end();
++row_j)
{
*row_j = *row_j * row_i->pivot() - *row_i * row_j->pivot();
}
/* the best we can do towards true row echelon form is reduce
* the leading coefficient by the row's GCD
*/
// *row_i /= gcd(*row_i);
}
return a;
}

辅助类的接口(interface),尚未经过常量正确性和使 C++ 工作的其他必要细节的审查。

template <typename _M>
class RowWithPivot {
public:
typedef _M::RowXpr Wrapped;
typedef _M::Scalar Scalar;

RowWithPivot(_M& matrix, size_t row);

Wrapped base();
operator Wrapped();

void swap(RowWithPivot& other);

int cmp(RowWithPivot& other) const;
bool operator <(RowWithPivot& other) const;

// returns the index of the first non-zero scalar
// (best to cache this)
int pivot_index() const;
// returns first non-zero scalar, or 0 if none
Scalar pivot() const;
};

template <typename _M, typename _R = RowWithPivot<_M> >
class RowMajor {
public:
typedef _R value_type;

RowMajor(_M& matrix);

operator _M&();
_M& base();

value_type operator[](size_t i);

class iterator {
public:
// standard random access iterator
...
};

iterator begin();
iterator end();
};

关于c++ - 如何在 C++ 中计算列随机矩阵的特征向量,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4357525/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com