C++:图的最短路径问题

一、简介

        在非网图中,最短路径是指两顶点之间经历的边数最少的路径。在网图中,最短路径是指两顶点之间经历的边上权值之和最少的路径。

         路径上的第一个顶点称为源点,最后一个顶点称为终点

        最短路径问题是图的一个比较典型的应用问题。例如,给定某公路网的n个城市以及这些城市之间相通公路的距离,能否找到城市A到城市B之间一条距离最近的通路呢?如果将城市用顶点表示城市间的公路用边表示,公路的长度作为边的权值,这个问题就归结为在网图中求顶点A和顶点B的最短路径。

        最短路径问题常见的两种算法是Dijkstra算法Floyd算法

         

二、Dijkstra算法 

         Dijkstra算法是一种用于求解单源最短路径的经典算法。它可以在带权重的、有向或无向的图中找出从起始点所有其他节点的最短路径。该算法要求边的权重为非负数,因此不能处理带有负权重的图。

        其基本思想是通过贪心策略,每次选择未处理节点中距离起始点最近的节点,逐步找到最短路径。

        图采用邻接矩阵存储。

        Dijkstra算法主要包括以下步骤:

  1. 初始化

    • 创建一个数组 dist,用来记录从起点到各个节点的最短距离。初始时,起点到自身的距离为0,其它节点的距离设为正无穷大。
    • 创建一个bool数组 visited,用来记录节点是否已经被处理过。初始时,所有节点都未被处理过。
  2. 循环查找

    • 每次从未被处理过的节点中选出距离起点最近的节点 u,并标记为已处理。
    • 遍历所有与 u 相邻的节点 v,如果通过 u 到达 v 的路径更短,则更新 dist[v],并将 u 作为 v 的前驱节点。
  3. 重复

    • 重复以上步骤,直到所有节点都被处理或找到的最短路径已经不能再更新。
  4. 终止

    • 所有节点被标记为已处理,或者没有可以继续处理的节点时,算法终止。此时,数组 dist 中存放的即是起点到各节点的最短距离。

 

#include <iostream>
using namespace std;
#include <vector>//功能:通过Kijkstra算法求最短路径
//参数:输入起点src、顶点数vertex_num、邻接矩阵edge
vector<int> kijkstra(int src, int vertexNum, vector<vector<int>> edge)
{vector<int> dist(vertexNum, INT_MAX);//存放起点到各顶点的最近距离,初始为最大值INT_MAXvector<bool> visited(vertexNum, false);//存放各顶点的访问状态,初始为未访问falsefor (int i = 0; i < vertexNum; i++)       //更新起点到各顶点的距离{dist[i] = edge[src][i];}dist[src] = 0;                            //起点到起点的距离为0visited[src] = true;                      //将起点标记为已访问//遍历所有节点,更新起点到各节点的最小距离(更新dist[]数组),最多执行(顶点数-1)次即可,实际上(顶点数-2)应该够了for (int j = 0; j < vertexNum-1; j++){int min(INT_MAX), u(-1);              //存储未访问节点中距离起点最近的点的值和下标for (int k = 0; k < vertexNum; k++)   //遍历所有节点{if (!visited[k] && dist[k] < min) //若该节点未被访问且起点到该点的距离较近{min = dist[k];u = k;                        //u为未访问的点中距离起点最近的点}}if (u == -1)                          //若没有未访问的点或者没有连通的点{return dist;                      //返回结果,包含起点到各点的最短距离}visited[u] = true;                   //将最近的这个顶点标记为已访问for (int n = 0; n < vertexNum; n++)  //更新起点到各顶点的距离,使起点到各顶点的距离最近{if (dist[n] > dist[u] + edge[u][n] && !visited[n]) {                                //若从起点到u再到n的距离比直接从起点到n的距离还小dist[n] = dist[u] + edge[u][n];}}}return dist;                             //返回结果,包含起点到各点的最短距离
}int main()
{//设置起点为0,顶点数为5int src(0),dest(4), vertex_num(5);//图的邻接矩阵,INT_MAX表示两点之间无边,0表示同一个点之间的距离vector<vector<int>> edge = {{0      , 10     , 3      , INT_MAX, INT_MAX},{INT_MAX, 0      , 1      , 2      , INT_MAX},{INT_MAX, 4      , 0      , 8      , 2      },{INT_MAX, INT_MAX, INT_MAX, 0      , 7      },{INT_MAX, INT_MAX, INT_MAX, 9      , 0      }};//存放起点到各点的最短距离vector<int> dist;//调用Kijkstra算法求最短路径dist = kijkstra(src, vertex_num, edge);//输入起点到终点的最短距离cout << dist[dest];return 0;
}

 

三、Floyd算法

        Floyd算法是一种用于求解图中任意两个顶点之间最短路径的算法。它适用于带权有向图和无向图,能够处理负权边(但不支持负权回路)。

        Floyd算法的核心思想是利用动态规划的原则,通过对图中每对顶点之间的路径进行更新,逐步找到所有顶点对之间的最短路径。算法使用一个二维数组 dist,其中 dist[i][j] 表示从顶点 i 到顶点 j 的最短路径的权重。  

        同Dijkstra算法,Floyd算法适用于图的邻接矩阵存储结构。

        以下是Floyd算法的具体步骤:

  1. 输入:图的顶点数vertexNum 和边的列表(每条边包含起始顶点、结束顶点及权重)。
  2. 初始化距离矩阵 dist
    • 对于每个顶点 i,设置 dist[i][i] = 0
    • 对于每条边 (u, v, w),设置 dist[u][v] = w
    • 其余的 dist[i][j] 设置为INT_MAX。
  3. 动态规划更新
    • 使用三重循环遍历所有顶点组合 (i, j, k),进行路径更新。
  4. 输出:最短路径矩阵 dist

 

#include <iostream>
using namespace std;
#include <vector>//输入参数:起点src、终点dest、顶点数vertexNum,邻接矩阵edge
void Floyd(int src,int dest,int vertexNum, vector<vector<int>> edge)
{vector<vector<int>> dist(vertexNum, vector<int>(vertexNum, INT_MAX));//最短距离矩阵,存储两点之间的最短距离for (int i = 0; i < vertexNum; i++){for (int j = 0; j < vertexNum; j++){if (i == j){dist[i][j] = 0;          //自己到自己的距离为0}else if (edge[i][j] != 0){dist[i][j] = edge[i][j]; //设置边的权重}}}//Floyd算法for (int k = 0; k < vertexNum; k++){for (int i = 0; i < vertexNum; i++){for (int j = 0; j < vertexNum; j++){if (dist[i][k] != INT_MAX && dist[k][j] != INT_MAX)//判断i到k,k到j是否通{dist[i][j] = min(dist[i][j], dist[i][k] + dist[k][j]);//看是i直接到j更近,还是i先经过k再到j更近}}}}//输出结果cout << dist[src][dest] << endl;
}int main()
{// 顶点数量int vertex_num = 4; // 图的邻接矩阵vector<vector<int>> edge = {{0, 5, 0, 10},{0, 0, 3, 0 },{0, 0, 0, 1 },{0, 0, 0, 0 }};//调用Floyd算法求最短路径Floyd(0, 3, vertex_num, edge);return 0;
}

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

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

相关文章

VGG原理与实战

VGG网络结构 这也更好的块状结构,256个卷积核 卷积就是我们的一个特征图啊往往都会缩小 &#xff0c;然后的话但它通道不会变.卷积一般是使用我们的通道C变大,磁化但是它的通道就是我们那个H和W一般都会变小.下采样的意思就是使分辨率变小 vgg—block内的卷积层都是同结构的意…

Kubernetes资源详解

华子目录 1.Kubernetes中的资源1.1资源管理介绍1.2资源管理方式1.2.1命令式对象管理1.2.2kubectl常见command命令1.2.3资源类型1.2.4常用资源类型 基本命令示例运行和调试命令示例高级命令示例总结 其他命令示例 1.Kubernetes中的资源 1.1资源管理介绍 在kubernetes中&#xf…

Nacos理论知识+应用案例+高级特性剖析

一、理论知识 Nacos功能 Nacos常用于注册中心、配置中心 Nacos关键特性 1、服务发现和服务健康监测 nacos作为服务注册中心可用于服务发现,并支持传输层&#xff08;TCP&#xff09;和应用层(HTTP&#xff09;的健康检查&#xff0c;并提供了agent上报和nacos server端主动…

Transformer架构概述(二)

目录 1. Transformer架构概述 1.1 《Attention is All You Need》论文概述 1.2 Transformer的模块组成 1.3 Encoder 和 Decoder 的区别与联系 2. Transformer的并行计算效率相对于RNN的提升 2.1 RNN中的顺序处理问题 2.2 Transformer中的并行化优势 3. Self-Attention机…

Pikachu-PHP反序列化

从后端代码可以看出&#xff0c;拿到序列化后的字符串&#xff0c;直接做反序列化&#xff1b;并且在前端做了展示&#xff1b; 如果虚拟化后的字符串&#xff0c;包含alert 内容&#xff0c;反序列化后&#xff0c;就会弹出窗口 O:1:"S":1:{s:4:"test";s…

OpenJudge | 置换选择排序

总时间限制: 1000ms 内存限制: 65536kB 描述 给定初始整数顺串&#xff0c;以及大小固定并且初始元素已知的二叉最小堆&#xff08;为完全二叉树或类似完全二叉树&#xff0c;且父元素键值总小于等于任何一个子结点的键值&#xff09;&#xff0c;要求利用堆实现置换选择排序&a…

Gralloc图形缓冲的分配过程

广告 首先帮我朋友打个广告 我们一起在运营一个视频号 感兴趣的可以帮忙点击右边这个小铃铛 铃铛 序 其实越往底下走在很多人嘴里就会变得很玄乎&#xff0c;变得不可思议&#xff0c;这里的gralloc就是一个native service&#xff0c;只是分装了一些调用接口&#xff0c;上…

Pikachu-目录遍历

目录遍历&#xff0c;跟不安全文件上传下载有差不多&#xff1b; 访问 jarheads.php 、truman.php 都是通过 get 请求&#xff0c;往title 参数传参&#xff1b; 在后台&#xff0c;可以看到 jarheads.php 、truman.php所在目录&#xff1a; /var/www/html/vul/dir/soup 图片…

传感器模块编程实践(二)W5500 SPI转以太网模块简介及驱动源码

文章目录 一.概要二.W5500芯片介绍W5500通讯协议介绍 三.W5500模块介绍四.W5500模块原理图五.W5500以太网模通讯实验六.CubeMX工程源代码下载七.小结 一.概要 我们介绍过单片机的以太网系统一般是由&#xff1a;单片机MACPHYRJ45。有些单片机比如STM32F407VET6芯片内部自带MAC…

windows下载Redis

1.下载地址 Releases tporadowski/redis GitHub 下载后&#xff0c;将压缩包解压到你的文件夹即可。&#xff08;此时&#xff0c;redis已经完成安装&#xff09; 2.使用 2.1双击redis.server.exe即可启动&#xff08;启动redis服务端&#xff09;&#xff08;或者在当前目…

超声波清洗机什么牌子值得入手?推荐四款入手不亏的眼镜清洗机

在当今这个注重细节完美的时代&#xff0c;超声波清洗机凭借其卓越的清洁效率、深层渗透力及细腻的清洗效果&#xff0c;迅速赢得了家庭与专业场景的青睐。无论是精细的珠宝、眼镜框&#xff0c;还是金属装饰品、电子设备乃至医疗器具&#xff0c;超声波技术都能精准祛除隐秘处…

0110 Redis缓存的更新策略

在很多高并发的场景如秒杀系统&#xff0c;QPS会瞬时暴增&#xff0c;如果采用直接读写数据库&#xff08;如MySQL&#xff09;的方式&#xff0c;很可能会将数据库打垮。因此这种场景需要引入Redis做缓存&#xff0c;应对高并发的访问。但同时也会引入新的风险&#xff0c;最常…

数据结构——List接口

文章目录 一、什么是List&#xff1f;二、常见接口介绍三、List的使用总结 一、什么是List&#xff1f; 在集合框架中&#xff0c;List是一个接口&#xff0c;通过其源码&#xff0c;我们可以清楚看到其继承了Collection。 Collection 也是一个接口&#xff0c;该接口中规范了后…

华为 HCIP-Datacom H12-821 题库 (31)

&#x1f423;博客最下方微信公众号回复题库,领取题库和教学资源 &#x1f424;诚挚欢迎IT交流有兴趣的公众号回复交流群 &#x1f998;公众号会持续更新网络小知识&#x1f63c; 1. 默认情况下&#xff0c;IS-IS Level-1-2 路由器会将 Level-2 区域的明细路由信息发布到Lev…

Python入门--函数

目录 1. 函数介绍 2. 函数的定义 3. 函数的参数 4. 函数的返回值 5. 函数说明文档 6. 函数的嵌套调用 7. 函数的作用域 (1). 局部变量 (2). 全局变量 (3). global关键字 1. 函数介绍 函数&#xff1a;是组织好的&#xff0c;可重复使用的&#xff0c;用来实现特定功能…

YOLO-V7 二元分类器

在评估二元分类器性能时&#xff0c;TP、FP、TN和FN是四个核心指标&#xff0c;它们分别代表真阳性、假阳性、真阴性和假阴性。以下是这些指标的定义、计算方法以及在实际应用中的意义&#xff1a; 定义 TP&#xff08;真阳性&#xff09;&#xff1a;模型正确预测为正类且实…

Yocto - 使用Yocto开发嵌入式Linux系统_06 掌握Bitbake工具

Grasping the BitBake Tool 在上一章中&#xff0c;我们了解了元数据、元数据集合概念以及 conf/layer.conf 的重要性。在本章中&#xff0c;我们将更深入地研究元数据&#xff0c;了解配方如何相互依赖&#xff0c;并了解 BitBake 如何处理依赖关系。 In the previous chapter…

k8s 中微服务之 MetailLB 搭配 ingress-nginx 实现七层负载

目录 1 MetailLB 搭建 1.1 MetalLB 的作用和原理 1.2 MetalLB功能 1.3 部署 MetalLB 1.3.1 创建deployment控制器和创建一个服务 1.3.2 下载MealLB清单文件 1.3.3 使用 docker 对镜像进行拉取 1.3.4 将镜像上传至私人仓库 1.3.5 将官方仓库地址修改为本地私人地址 1.3.6 运行清…

【路径规划】多机器人路径规划

摘要 多机器人路径规划在现代自动化、仓储管理及智能交通系统中有着广泛的应用。本文提出了一种基于A*算法的多机器人路径规划方法&#xff0c;旨在解决多机器人在同一环境中的路径冲突问题。通过采用启发式搜索和路径优化策略&#xff0c;机器人能够在保持避障的前提下实现最…

Middleware---RocketMQ

RocketMQ是一个开源的分布式消息中间件。它是一种 低延迟、高可用、高可靠、高并发 的消息队列系统&#xff0c;用于在分布式系统中进行异步通信。 RocketMQ架构模型 Producer Group&#xff1a;消息生产者组&#xff0c;负责发送消息。 Broker&#xff1a;存储消息的服务节…