CGS与MGS的矩阵正交化-C语言实现

格拉姆-施密特正交化和改进的格拉姆-施密特正交化

格拉姆-施密特正交化CGS

数学公式

代码实现:

过程版

矩阵运算实现的难点在于每次运算都是一个向量,需要for循环进行,会带来运算时在代码中的复杂,进而难以理解代码的过程

cpp 复制代码
Q矩阵是{e...}  R矩阵是上三角矩阵 
Q[0] = A[0];   // -> b1 = a1
R[0][0] = norm(Q[0], m); //-> ||b1||
Q[0] =A[0] / R[0][0];    //-> b1/||b1||

先求 R[1][2] = dotProduct(A[i], Q[j], m);

再用 Q[2] = A[2]; //这样可以使用迭代去求b2

cpp 复制代码
R[j][i] = dotProduct(A[i], Q[j], m); // -> a2Te1

Q[i][k] = A[i][k];   //方便下一般实现循环递归减
Q[i][k] -=  R[j][i]*Q[j][k];   //用于递归实现 bn = an - anTen-1en-1-...

R[i][i] = norm(Q[i], m);// -> ||b1||
Q[i][k] /= R[i][i];    //-> b1/||b1||

纯净版

cpp 复制代码
void cgs(double** A, int m, int n, double** Q, double** R) {
   
    if (n>=1)
    {
        R[0][0] = norm(A[0], m);
        for (int k = 0; k < m; k++) {
            Q[0][k] =A[0][k]/ R[0][0];
        }
    }
    for (int i = 1; i < n; i++) {
        for (int j = 0; j < i; j++) {
            R[j][i] = dotProduct(A[i], Q[j], m);
        }
        for (int k = 0; k < m; k++)
        {
           Q[i][k] = A[i][k];
        }
        for (int j = 0; j < i; j++) {
            for (int k = 0; k < m; k++) {
                Q[i][k] -=  R[j][i]*Q[j][k];
            }
        }
        R[i][i] = norm(Q[i], m);
        for (int k = 0; k < m; k++) {
            Q[i][k] /= R[i][i];
        }
    }
}

改进的格拉姆-施密特正交化MGS

数学公式

在于先求e 每次都对全部b进行运算

代码实现

过程版

for (int i = 0; i < m; i++)

{

Q[i]= A[i];

}

cpp 复制代码
for (int i = 0; i < m; i++)
{
   for (int j = 0; j < n; j++)
   {
       Q[i][j] = A[i][j];
   }
}
cpp 复制代码
R[i][i] = norm(Q[i], m);
for (int k = 0; k < m; k++) {
      Q[i][k] /= R[i][i];
}

先求

cpp 复制代码
 R[i][j] = dotProduct(Q[i], Q[j], m);

对每一个b仅需运算 每一个循环都会少计算b1,b2,b3..bn所以j = i+1

cpp 复制代码
 for (int j = i + 1; j < n; j++) {
     for (int k = 0; k < m; k++) {
        Q[j][k] -= Q[i][k] * R[i][j];
     }
 }

纯净版

复制代码
void mgs(double** A, int m, int n, double** Q, double** R) {
    for (int i = 0; i < m; i++)
    {
        for (int j = 0; j < n; j++)
        {
           Q[i][j] = A[i][j];
        }
    }
    for (int i = 0; i < n; i++) {
        R[i][i] = norm(Q[i], m);
        for (int k = 0; k < m; k++) {
            Q[i][k] /= R[i][i];
        }
        // 计算 R元素 b2Tei b3Tei...  Q[i]=e[i]
        for (int j = i + 1; j < n; j++) {
            R[i][j] = dotProduct(Q[i], Q[j], m);
        }
        // 更新 Q
        for (int j = i + 1; j < n; j++) {
            for (int k = 0; k < m; k++) {
                Q[j][k] -= Q[i][k] * R[i][j];
            }
        }
    }
}

参考博客:

下文有博主有python实现

图解格拉姆-施密特正交化和改进的格拉姆-施密特正交化_matlab实现格拉姆-施密特正交化-CSDN博客

相关推荐
犯困的土子哥几秒前
C++:哈希表
c++·哈希算法
Code Warrior31 分钟前
【Linux】Socket 编程预备知识
linux·网络·c++
智者知已应修善业33 分钟前
【c语言蓝桥杯计算卡片题】2023-2-12
c语言·c++·经验分享·笔记·算法·蓝桥杯
littlepeanut.top33 分钟前
C++中将FlatBuffers序列化为JSON
开发语言·c++·json·flatbuffers
hansang_IR39 分钟前
【题解】洛谷 P2330 [SCOI2005] 繁忙的都市 [生成树]
c++·算法·最小生成树
Croa-vo1 小时前
PayPal OA 全流程复盘|题型体验 + 成绩反馈 + 通关经验
数据结构·经验分享·算法·面试·职场和发展
是苏浙1 小时前
零基础入门C语言之贪吃蛇的实现
c语言·开发语言·数据结构
AndrewHZ2 小时前
【图像处理基石】 怎么让图片变成波普风?
图像处理·算法·计算机视觉·风格迁移·cv
无极小卒2 小时前
如何在三维空间中生成任意方向的矩形内部点位坐标
开发语言·算法·c#
FMRbpm2 小时前
链表中出现的问题
数据结构·c++·算法·链表·新手入门