高斯消元法及其C++实现

news2025/2/12 15:34:04

深入浅出高斯消元法及其C++实现

本文章代码由博主编写但是文章由ChatGPT-o1-mini生成
博客食用更佳

在计算机算法竞赛中,线性方程组的求解是一个常见且基础的问题。高斯消元法作为一种经典的算法,因其高效和直观的特性,广泛应用于各种编程竞赛和实际问题中。本文将通过一个具体的C++实现,深入浅出地讲解高斯消元法的核心概念、实现细节以及如何应对实际编程中的挑战。

一、问题背景

高斯消元法(Gaussian Elimination)是一种用于求解线性方程组的算法。给定一个由 n n n个方程组成的线性方程组,每个方程包含 n n n个未知数,我们希望通过高斯消元法找到这些未知数的解。

题目描述

我们需要编写一个程序,输入一个 n × ( n + 1 ) n \times (n+1) n×(n+1)的矩阵,其中前 n n n列代表方程组的系数,最后一列代表常数项。程序需要输出方程组的唯一解,如果方程组无解或有无穷多解,则输出No Solution

输入输出示例

输入示例:

3
1 3 4 5
1 4 7 3
9 3 2 2

输出示例:

-0.97
5.18
-2.39

二、高斯消元法基础

1. 基本思想

高斯消元法通过一系列行变换,将原始的线性方程组转化为上三角矩阵形式。具体步骤如下:

  1. 选主元(Pivoting): 对于每一列,选择绝对值最大的元素作为主元,以提高数值稳定性。
  2. 消元(Elimination): 利用主元,将其下方所有元素归零。
  3. 回代(Back Substitution): 从上到下求解未知数。

2. 处理特殊情况

在实际应用中,可能会遇到以下几种情况:

  • 无解: 方程组中出现矛盾,如 0 x + 0 y + 0 z = 1 0x + 0y + 0z = 1 0x+0y+0z=1
  • 有无穷多解: 自由变量存在,方程组没有唯一解。
  • 唯一解: 每个未知数都有确定的值。

三、C++实现解析

让我们逐步解析提供的C++代码,理解高斯消元法在代码中的具体实现。

1. 代码结构概述

#include <cmath>
#include <cstdint>
#include <iomanip>
#include <ios>
#include <iostream>
#include <utility>

using ll = int64_t;

template<class T>
T read(){
    T t;
    std::cin>>t;
    return t;
}

constexpr ll maxn = 100;
ll n;
double im[maxn+5][maxn+5], tmp[maxn+5];

void cp(double*f,double*t){
    for(ll i=1;i<=n+1;i++){
        t[i]=f[i];
    }
}

void sp(double*f,double*t){
    for(ll i=1;i<=n+1;i++){
        std::swap(f[i],t[i]);
    }
}

int main(){
    std::cin>>n;
    for(ll i=1;i<=n;i++){
        for(ll j=1;j<=n+1;j++){
            std::cin>>im[i][j];
        }
    }
    for(ll i=1;i<=n;i++){
        for(ll k=i;k<=n;k++){
            if(im[k][i]!=0){
                if(k!=i){
                    sp(im[k],im[i]);
                }
                break;
            }
        }
        double fact = im[i][i];
        for(ll j=1;j<=n+1;j++){
            im[i][j]/=fact;
        }
        for(ll k=1;k<=n;k++){
            if(k==i)continue;
            double fact = im[k][i];
            for(ll j=1;j<=n+1;j++){
                im[k][j]-=fact*im[i][j];
            }
        }
    }

    if([]()->bool{
        for(ll i=1;i<=n;i++){
            bool isAllZero=true;
            for(ll j=1;j<=n;j++){
                if(im[i][j]!=0){
                    isAllZero=false;
                }
            }
            if(isAllZero){
                return false;
            }
        }
        for(ll i=1;i<=n;i++){
            if(std::isnan(im[i][n+1])){
                return false;
            }
        }
        return true;
    }()){
        std::cout<<std::fixed<<std::setprecision(2);
        for(ll i=1;i<=n;i++){
            std::cout<<im[i][n+1]<<'\n';
        }
    }else{
        std::cout<<"No Solution\n";
    }
}

2. 代码详解

2.1 数据输入
std::cin>>n;
for(ll i=1;i<=n;i++){
    for(ll j=1;j<=n+1;j++){
        std::cin>>im[i][j];
    }
}
  • 首先读取方程组的规模 n n n
  • 然后逐行读取矩阵数据,其中每一行包含 n n n个系数和1个常数项。
2.2 主元选择与行交换
for(ll i=1;i<=n;i++){
    for(ll k=i;k<=n;k++){
        if(im[k][i]!=0){
            if(k!=i){
                sp(im[k],im[i]);
            }
            break;
        }
    }
    // ...
}
  • 对于第 i i i个未知数,选择列 i i i中第一个非零元素作为主元。
  • 如果主元不在当前行,则交换行以将主元移至当前行。

函数sp的定义:

void sp(double*f,double*t){
    for(ll i=1;i<=n+1;i++){
        std::swap(f[i],t[i]);
    }
}
  • sp函数用于交换两行的数据。
2.3 主元归一化
double fact = im[i][i];
for(ll j=1;j<=n+1;j++){
    im[i][j]/=fact;
}
  • 将主元所在的整行都除以主元,使得主元变为1,简化后续消元过程。
2.4 消元过程
for(ll k=1;k<=n;k++){
    if(k==i)continue;
    double fact = im[k][i];
    for(ll j=1;j<=n+1;j++){
        im[k][j]-=fact*im[i][j];
    }
}
  • 对于除了当前主行之外的每一行,消去所在列的元素,使得每列只有主元为1,其他元素为0。
2.5 判断解的唯一性
if([]()->bool{
    for(ll i=1;i<=n;i++){
        bool isAllZero=true;
        for(ll j=1;j<=n;j++){
            if(im[i][j]!=0){
                isAllZero=false;
            }
        }
        if(isAllZero){
            return false;
        }
    }
    for(ll i=1;i<=n;i++){
        if(std::isnan(im[i][n+1])){
            return false;
        }
    }
    return true;
}()){
    std::cout<<std::fixed<<std::setprecision(2);
    for(ll i=1;i<=n;i++){
        std::cout<<im[i][n+1]<<'\n';
    }
}else{
    std::cout<<"No Solution\n";
}
  • 通过一个lambda表达式判断方程组是否有唯一解。
  • 检查所有主元行:
    • 如果存在某一行系数全为0,但常数项不为0,说明方程组无解。
  • 检查解的合理性:
    • 如果某个解为NaN(非数),说明方程组无唯一解。
  • 如果满足唯一解的条件,则输出结果,保留两位小数。
  • 否则,输出No Solution

3. 代码中的关键点

3.1 数组下标从1开始

在这段代码中,数组的下标从1开始,而不是C++中常见的从0开始。这种设计可能是为了更贴近数学表达的习惯,使得代码更容易理解。

3.2 行交换与主元选择

选择主元并交换行是高斯消元法中的关键步骤,能够有效避免在消元过程中遇到除以0的情况,并提高算法的数值稳定性。

3.3 消元与归一化

通过对主元所在行进行归一化处理,使得主元为1,然后利用主元消去其他行同一列的元素,最终将矩阵转化为简化的上三角矩阵或单位矩阵。

3.4 解的判断

判断方程组是否有唯一解是高斯消元法中的重要部分。通过检查消元后的矩阵,可以确定方程组的解的性质。

4. 代码优化建议(不修改代码的前提下)

尽管用户要求不修改代码,但从学习的角度,可以提出一些优化建议:

  • 使用零近似判断: 由于浮点数的精度问题,直接比较im[k][i] != 0可能不可靠。可以引入一个极小的阈值,例如1e-9,判断abs(im[k][i]) > 1e-9
  • 更智能的主元选择: 当前代码选择的是首个非零元素作为主元,可以改为选择绝对值最大的元素以提高数值稳定性。
  • 使用vector替代固定大小的数组: 虽然本题数据范围较小,但在更复杂的场景下,vector提供了更大的灵活性。

四、实例解析

让我们通过给定的输入示例,手动模拟高斯消元法的过程,帮助理解代码的执行流程。

输入:

3
1 3 4 5
1 4 7 3
9 3 2 2

步骤:

  1. 初始增广矩阵:

    1  3  4 | 5
    1  4  7 | 3
    9  3  2 | 2
    
  2. 第1步:选择第1列的主元

    • 最大绝对值在第3行,元素为9。
    • 交换第1行与第3行。
    9  3  2 | 2
    1  4  7 | 3
    1  3  4 | 5
    
  3. 第2步:归一化主元行

    • 主元为9,将整个第1行除以9。
    1  0.3333  0.2222 | 0.2222
    1  4       7      | 3
    1  3       4      | 5
    
  4. 第3步:消元

    • 消去第2行和第3行的第1列元素。

    • 第2行减第1行:

      0  3.6667  6.7778 | 2.7778
      
    • 第3行减第1行:

      0  2.6667  3.7778 | 4.7778
      
  5. 重复以上步骤,直到矩阵达到单位矩阵形式。

最终得到:

1  0  0 | -0.97
0  1  0 | 5.18
0  0  1 | -2.39

即:

x = -0.97
y = 5.18
z = -2.39

五、总结

本文通过一个具体的C++实现,详细解析了高斯消元法的基本原理和编程实现细节。高斯消元法作为求解线性方程组的经典算法,具有广泛的应用场景和重要的理论价值。在竞赛编程中,理解并掌握高斯消元法的实现,不仅能够帮助解决相关问题,还能提高对线性代数基础知识的理解。

在实际编程中,注意数值稳定性和特殊情况的处理至关重要。通过合理的主元选择和精确的浮点数运算,可以有效避免算法中的潜在问题。同时,合理的数据结构和优化技巧也能提升代码的性能和可读性。

希望通过本文,读者能够对高斯消元法有更深入的理解,并能在编程竞赛中灵活运用这一强大的工具。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2296910.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

DeepSeek AI R1推理大模型API集成文档

DeepSeek AI R1推理大模型API集成文档 引言 随着自然语言处理技术的飞速发展&#xff0c;大语言模型在各行各业的应用日益广泛。DeepSeek R1作为一款高性能、开源的大语言模型&#xff0c;凭借其强大的文本生成能力、高效的推理性能和灵活的接口设计&#xff0c;吸引了大量开发…

【算法-动态规划】、魔法卷轴: 两次清零机会整个数组最大累加和

【算法-动态规划】、魔法卷轴: 两次清零机会整个数组最大累加和 文章目录 一、dp1.1 题意理解1.2 整体思路1.3 具体思路1.4 代码 二、多语言解法 一、dp 1.1 题意理解 nums 数组, 有正负0, 使用最多两次魔法卷轴, 希望使数组整体的累加和尽可能大. 求尽可能大的累加和 其实就…

【R】Dijkstra算法求最短路径

使用R语言实现Dijkstra算法求最短路径 求点2、3、4、5、6、7到点1的最短距离和路径 1.设置data&#xff0c;存放有向图信息 data中每个点所在的行序号为起始点序号&#xff0c;列为终点序号。 比如&#xff1a;值4的坐标为(1,2)即点1到点2距离为4&#xff1b;值8的坐标为(6,7)…

深入浅出:探索 DeepSeek 的强大功能与应用

深入浅出&#xff1a;探索 DeepSeek 的强大功能与应用 在人工智能技术飞速发展的今天&#xff0c;自然语言处理&#xff08;NLP&#xff09;作为其重要分支&#xff0c;正逐渐渗透到我们生活的方方面面。DeepSeek 作为一款功能强大的 NLP 工具&#xff0c;凭借其易用性和高效性…

西门子S7-200 PLC串口PPI转以太网通讯的模块链接方式

项目背景 某汽车零部件生产车间有30台自动化生产设备&#xff0c;控制系统采用西门子S7-200系列的CPU226。此前&#xff0c;设备的一个通讯端口用于和变频器进行自由口通讯&#xff0c;另一个通讯端口连接着一台昆仑通态触摸屏作为人机界面。车间计划进行智能化升级&#xff…

win10向windows server服务器传输文件

win10向windows server服务器传输文件 遇到无法直接拖动文件进行传输时 解决方案&#xff1a; 1.点击显示选项 2.点击本地资源-详细信息 3.在窗口中选择你需要共享的磁盘 4.然后远程连接到Windows server服务器 5.登录Windows server服务器后&#xff0c;在此电脑下就能看…

git服务器搭建,gitea服务搭建,使用systemclt管理服务

文章目录 页面展示使用二进制文件安装git服务下载选择架构使用wget下载安装 验证 GPG 签名服务器设置准备环境创建systemctl文件 备份与恢复备份命令 (dump)恢复命令 (restore) 页面展示 使用二进制文件安装git服务 所有打包的二进制程序均包含 SQLite&#xff0c;MySQL 和 Po…

Mybatis快速入门与核心知识总结

Mybatis 1. 实体类&#xff08;Entity Class&#xff09;1.1 实体类的定义1.2 简化编写1.2.1 Data1.2.2 AllArgsConstructor1.2.3 NoArgsConstructor 2. 创建 Mapper 接口2.1 Param2.2 #{} 占位符2.3 SQL 预编译 3. 配置 MyBatis XML 映射文件&#xff08;可选&#xff09;3.1 …

用docker在本地用open-webui部署网页版deepseek

前置条件 用Ollama在本地CMD窗口运行deepseek大模型-CSDN博客文章浏览阅读109次&#xff0c;点赞5次&#xff0c;收藏2次。首次运行需要下载deepseek的大模型包&#xff08;大约5GB&#xff0c;根据本地网速的不同在半个小时到几个小时之间下载完成&#xff09; &#xff0c;并…

2025.2.8——一、[护网杯 2018]easy_tornado tornado模板注入

题目来源&#xff1a;BUUCTF [护网杯 2018]easy_tornado 目录 一、打开靶机&#xff0c;整理信息 二、解题思路 step 1&#xff1a;分析已知信息 step 2&#xff1a;目标——找到cookie_secret step 3&#xff1a;构造payload 三、小结 一、打开靶机&#xff0c;整理信…

前端实现在PDF上添加标注(1)

前段时间接到一个需求&#xff0c;用户希望网页上预览PDF&#xff0c;同时能在PDF上添加文字&#xff0c;划线&#xff0c;箭头和用矩形框选的标注&#xff0c;另外还需要对已有的标注进行修改&#xff0c;删除。 期初在互联网上一通搜索&#xff0c;对这个需求来讲发现了两个问…

GitCode 助力 Easy-Es,革新 Elasticsearch 开发体验

项目仓库&#xff08;点击阅读原文链接可直达&#xff09; https://gitcode.com/dromara/easy-es 项目背景&#xff1a;填补 Elasticsearch ORM 框架空白 在 Java 开发领域&#xff0c;Excel 和 Elasticsearch 的代码编写难度一直名列前茅&#xff0c;尤其是 Elasticsearch&a…

EF Core中实现值对象

目录 值对象优点 值对象的需求 值类型的实现 值类型GEO的实现 值类型MultilingualString的实现 案例&#xff1a;构建表达式树&#xff0c;简化值对象的比较 值对象优点 把有紧密关系的属性打包为一个类型把领域知识放到类的定义中 class shangjia {long id;string nam…

《从入门到精通:蓝桥杯编程大赛知识点全攻略》(十一)-回文日期、移动距离、日期问题

前言 在这篇博客中&#xff0c;我们将通过模拟的方法来解决三道经典的算法题&#xff1a;回文日期、移动距离和日期问题。这些题目不仅考察了我们的基础编程能力&#xff0c;还挑战了我们对日期处理和数学推理的理解。通过模拟算法&#xff0c;我们能够深入探索每个问题的核心…

Docker Compose介绍及安装使用MongoDB数据库详解

在现代容器化应用部署中&#xff0c;Docker Compose是一种非常实用的工具&#xff0c;它允许我们通过一个docker-compose.yml文件来定义和运行多容器应用程序。然而&#xff0c;除了Docker之外&#xff0c;Podman也提供了类似的工具——Podman Compose&#xff0c;它允许我们在…

11.swagger使用

菜单位置 未登录接口会返回401 登录的token存储的位置 配置文件swagger配置中将/dev-api修改/

java高级知识之集合

前言 集合是java开发中的重点内容&#xff0c;需要掌握的东西很多&#xff0c;面试中可问的东西很多&#xff0c;无论是深度还是广度。集合框架中Collection对应的实现类如下所示&#xff0c;这些都是要完全掌握&#xff0c;一个可以分为三大类List集合、Set‘集合以及Map集合…

deepseek + kimi 高效生成PPT

1.在deepseek中生成ppt大纲 2.将大纲复制到kimi中生成PPT kimi&#xff1a;https://kimi.moonshot.cn/

好好说话:深度学习扫盲

大创项目是和目标检测算法YOLO相关的&#xff0c;浅浅了解了一些有关深度学习的知识。在这里根据本人的理解做一些梳理。 深度学习是什么&#xff1f; 之前经常听到AI&#xff0c;机器学习&#xff0c;深度学习这三个概念&#xff0c;但是对于三者的区别一直很模糊。 AI&…

【愚公系列】《Python网络爬虫从入门到精通》001-初识网络爬虫

标题详情作者简介愚公搬代码头衔华为云特约编辑&#xff0c;华为云云享专家&#xff0c;华为开发者专家&#xff0c;华为产品云测专家&#xff0c;CSDN博客专家&#xff0c;CSDN商业化专家&#xff0c;阿里云专家博主&#xff0c;阿里云签约作者&#xff0c;腾讯云优秀博主&…