最短路径问题(dj和floyd算法)

2024-03-10 10:18

本文主要是介绍最短路径问题(dj和floyd算法),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 首先是dj算法,dj算法的思想是从开始结点开始,每次找到距离开始结点距离最近的结点newP,加入到集合K中,表示已经访问过,然后遍历newP的直接相邻的结点,如果从newP结点开始到直接相邻的结点 i 的距离Dis[newP]+c < Dis[i],那么就更新Dis[i],扫描完成所有newP的邻接结点,然后遍历所有的结点,找到未访问过的并且Dis[i]最小的结点 j ,设为新的 newP ,重复上述步骤,直到所有的结点都完成了访问。

设置一个Dis[i]矩阵,表示从1到该结点的距离,Dis[1] = 0, 开始时其它结点的Dis[1] = -1.从1开始进行循环。

如果要打印最短路径,可以设置一个vector<int> pre[N], 每个结点 i ,将在更新Dis[i]时,更新 pre[i].push_back(newP), 通过newP到达 i 是当前从1到达i 的最短路径的上一个结点,这样从后向前打印即可。

看下面的这个例子:

从1出发,找到最短的邻接点 3,访问;从 3 开始遍历 3 的邻接点, 更新5的Dis[5] = 7, 从未访问的结点2,4,5中找到最近的结点2,访问,然后是4访问,遍历4的邻接点,Dis[5]更新为6,最后访问5;完成;

问题是:给出N个地点,M个直接路径,直接路径包括两端地点和它们直接长度,找出1到N号地点的最短路径:

代码为:

#include<cstdio>
#include<vector>
#include<math.h>
#include<algorithm>
using namespace std;
#define N 101
struct E{int next;int c;
};
vector<E> edge[N];//edge[i]中的与元素是一个个的E类型,包含了直接相连号和边权
bool mark[N];//表示该结点是否已经加入到最短路径已知集合中
int Dis[N];//从开始结点到任意结点的最短路径长度度int main()
{int n,m;//n个地点,m个直接相连路径,求从1到n的最短路径while(scanf("%d%d", &n, &m) != EOF){if(n == 0 && m == 0)break;for(int i = 1; i <= n; i++)edge[i].clear();while(m--){int a,b,cost;scanf("%d%d%d", &a, &b, &cost);E temp;temp.next = b;temp.c = cost;edge[a].push_back(temp);temp.next = a;edge[b].push_back(temp);//无向边,所以两个点的邻接链表都要增加}for(int i = 1; i <= n; i++){Dis[i] = -1;mark[i] = false;}Dis[1] = 0;mark[1] = true;int newP = 1;//表示是上一个加入到最短路径的点for(int i = 1; i < n; i++)//循环n-1次,将所有的点都加入到集合中{for(int j = 0; j < edge[newP].size(); j++){int t = edge[newP][j].next;int ct = edge[newP][j].c;if(mark[t] == true)continue;if(Dis[t] == -1 || Dis[t] > Dis[newP] + ct)Dis[t] = Dis[newP] + ct;}int min = 100000000;for(int k = 1; k <= n; k++){if(mark[k] == true)continue;if(Dis[k] == -1)continue;if(min > Dis[k]){min = Dis[k];newP = k;}}mark[newP] = true;}printf("%d\n", Dis[n]);}return 0;
}

运行结果,最后一个例子是上面的图:

2. floyd算法求最短路径,这种算法要求事先得到所有直接相连结点间的距离,然后依次判断i,j之间是否存在一个中间结点,使得

ans[i][j] > ans[i][k] + ans[k][j]从而得到新的ans[i][j]的值,要对这个中间结点进行所有结点的遍历,不可达的要跳过,对所有ans[i][j]进行遍历n次,就可以得到所有结点间的最短距离,包括i, j之间包含多个中间结点的情况,不用深入研究,记住即可。注意设定ans[i][i] = 0,所以 ans[i][j] == ans[i][i] + ans[i][j],不会进行更新。

代码为:

#include<cstdio>
#include<iostream>
using namespace std;int ans[101][101]; //初始值为邻接矩阵,只有直接相邻的结点的距离才有意义,否则为-1
int main()
{int n, m;while(scanf("%d%d", &n, &m) != EOF){if(n == 0 && m == 0)break;for(int i = 1; i <= n; i++){for(int j = 1; j <= n; j++){ans[i][j] = -1;}ans[i][i] = 0; //自己到自己的路径长度设为0}while(m--){int a, b, c;scanf("%d%d%d", &a, &b, &c);ans[a][b] = ans[b][a] = c;//无向图}for(int k = 1; k <= n; k++)for(int i = 1; i <= n; i++)for(int j = 1; j <= n; j++){if(ans[i][k] == -1 || ans[k][j] == -1)continue;if(ans[i][j] == -1 || ans[i][k] + ans[k][j] < ans[i][j])ans[i][j] = ans[i][k] + ans[k][j];}printf("%d\n", ans[1][n]);}return 0;
}

运行结果为:

这篇关于最短路径问题(dj和floyd算法)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/793917

相关文章

Springboot3统一返回类设计全过程(从问题到实现)

《Springboot3统一返回类设计全过程(从问题到实现)》文章介绍了如何在SpringBoot3中设计一个统一返回类,以实现前后端接口返回格式的一致性,该类包含状态码、描述信息、业务数据和时间戳,... 目录Spring Boot 3 统一返回类设计:从问题到实现一、核心需求:统一返回类要解决什么问题?

maven异常Invalid bound statement(not found)的问题解决

《maven异常Invalidboundstatement(notfound)的问题解决》本文详细介绍了Maven项目中常见的Invalidboundstatement异常及其解决方案,文中通过... 目录Maven异常:Invalid bound statement (not found) 详解问题描述可

idea粘贴空格时显示NBSP的问题及解决方案

《idea粘贴空格时显示NBSP的问题及解决方案》在IDEA中粘贴代码时出现大量空格占位符NBSP,可以通过取消勾选AdvancedSettings中的相应选项来解决... 目录1、背景介绍2、解决办法3、处理完成总结1、背景介绍python在idehttp://www.chinasem.cna粘贴代码,出

SpringBoot整合Kafka启动失败的常见错误问题总结(推荐)

《SpringBoot整合Kafka启动失败的常见错误问题总结(推荐)》本文总结了SpringBoot项目整合Kafka启动失败的常见错误,包括Kafka服务器连接问题、序列化配置错误、依赖配置问题、... 目录一、Kafka服务器连接问题1. Kafka服务器无法连接2. 开发环境与生产环境网络不通二、序

SpringSecurity中的跨域问题处理方案

《SpringSecurity中的跨域问题处理方案》本文介绍了跨域资源共享(CORS)技术在JavaEE开发中的应用,详细讲解了CORS的工作原理,包括简单请求和非简单请求的处理方式,本文结合实例代码... 目录1.什么是CORS2.简单请求3.非简单请求4.Spring跨域解决方案4.1.@CrossOr

nacos服务无法注册到nacos服务中心问题及解决

《nacos服务无法注册到nacos服务中心问题及解决》本文详细描述了在Linux服务器上使用Tomcat启动Java程序时,服务无法注册到Nacos的排查过程,通过一系列排查步骤,发现问题出在Tom... 目录简介依赖异常情况排查断点调试原因解决NacosRegisterOnWar结果总结简介1、程序在

解决java.util.RandomAccessSubList cannot be cast to java.util.ArrayList错误的问题

《解决java.util.RandomAccessSubListcannotbecasttojava.util.ArrayList错误的问题》当你尝试将RandomAccessSubList... 目录Java.util.RandomAccessSubList cannot be cast to java.

Apache服务器IP自动跳转域名的问题及解决方案

《Apache服务器IP自动跳转域名的问题及解决方案》本教程将详细介绍如何通过Apache虚拟主机配置实现这一功能,并解决常见问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,... 目录​​问题背景​​解决方案​​方法 1:修改 httpd-vhosts.conf(推荐)​​步骤

java反序列化serialVersionUID不一致问题及解决

《java反序列化serialVersionUID不一致问题及解决》文章主要讨论了在Java中序列化和反序列化过程中遇到的问题,特别是当实体类的`serialVersionUID`发生变化或未设置时,... 目录前言一、序列化、反序列化二、解决方法总结前言serialVersionUID变化后,反序列化失

C++ 多态性实战之何时使用 virtual 和 override的问题解析

《C++多态性实战之何时使用virtual和override的问题解析》在面向对象编程中,多态是一个核心概念,很多开发者在遇到override编译错误时,不清楚是否需要将基类函数声明为virt... 目录C++ 多态性实战:何时使用 virtual 和 override?引言问题场景判断是否需要多态的三个关