详解最短路算法模板(dijkstra+floyd+spfa)

1.Floyd_Warshall算法

核心思路:d[i][j] = min{d[i][j], d[i][k] + d[k][j]}
从i到j有两种路径,经过k点或是不经过k点,所以我们枚举k即可求所有路的最短路。
适用范围:求任意两点间的最短路,可以有负权,可以是有向图可以是无向图,但是n必须在200以内

#include <iostream>
#include <stdio.h>
#include <string>
#include <string.h>
#include <queue>
#include <vector>
#define INF 0x3f3f3f3fusing namespace std;int main()
{int n, m, s, t;while(~scanf("%d%d", &n, &m)){vector<vector<int> > dis(n);//vector二维可变长数组for(int i = 0; i < n; i++){dis[i].resize(n, INF);//初始化设置dis[i]的长度,并用INF作为初始值dis[i][i] = 0;}for(int i = 0; i < m; i++)//输入边a,b两点的权值是x{int a, b, x;scanf("%d%d%d", &a, &b, &x);if(dis[a][b] > x)dis[a][b] = dis[b][a] = x;}scanf("%d%d", &s, &t);for(int k = 0; k < n; k++)for(int i = 0; i < n; i++)for(int j = 0; j < n; j++){if(dis[i][k] < INF && dis[k][j] < INF)dis[i][j] = min(dis[i][j], dis[i][k] + dis[k][j]);}if(dis[s][t] != INF)//可以求任意两点间最短路printf("%d\n", dis[s][t]);elseprintf("-1\n");}return 0;
}

2.Dijkstra算法

核心思路:
D(s, t) = {Vs … Vi … Vj … Vt}表示s到t的最短路,其中i和j是这条路径上的两个中间结点,那么D(i, j)必定是i到j的最短路,如果存在这样一条最短路D(s, t) = {Vs … Vi Vt},其中i和t是最短路上相邻的点,那么D(s, i) = {Vs … Vi} 必定是s到i的最短路。Dijkstra算法就是基于这样一个性质,通过最短路径长度递增,逐渐生成最短路。
适用情况:
正权图上的单元最短路,有向图无向图,从单个源点出发到所有结点的最短路

设起始点为s
清除所有点的标记
设dis[s] = 0,其他dis[i] = INF
循环n次
{ 在所有未标记的节点中,选出dis值最小的节点X给节点X标记对于从X出发可以到达的点y,更新dis[y] = min{dis[y], dis[x]+w(x,y)}
}

邻接矩阵实现:

void dijkstra(int s)//s是起点
{memset(dis, INF, sizeof(dis));memset(vis,0,sizeof(vis);vis[s] = 1;dis[s] = 0;for(int i = 1; i <= n; i++)//执行n-1轮{int min_dis = INF;int x;for(int j = 1; j <= n; j++)//寻找所有集合外的点到集合距离最小的点x{if(!vis[j] && min_dis > dis[j]){x = j;min_dis = dis[j];}}vis[x] = 1;//然后把X加入到最短路点集中for(int j = 1; j <= n; j++)//更新集合外点到集合的距离{if(!vis[j])dis[j] = min(dis[j], dis[x] + mapp[x][j]);//x到j的距离+dis[x]}}
}

优先队列优化版

#include <iostream>
#include <stdio.h>
#include <string>
#include <string.h>
#include <queue>
#include <vector>
#define INF 0x3f3f3f3fusing namespace std;
const int maxn = 105;
int dis[maxn], pre[maxn];struct Edge//边 
{int u, v, w;Edge() {};Edge(int uu, int vv, int ww): u(uu), v(vv), w(ww) {};
};vector<Edge> edges;//边数组 
vector<int> G[maxn];//存储每个节点对应的边的序号 void init(int nn)//清理 
{for(int i = 0; i <= nn; i++)G[i].clear();edges.clear();
}void AddEdge(int uu, int vv, int ww)//加边 
{edges.push_back(Edge(uu, vv, ww));int edgenum = edges.size();G[uu].push_back(edgenum - 1);
}struct node//优先队列优化,dis小的先出队 
{int u, d;node() {};node(int uu, int dd): u(uu), d(dd) {};friend bool operator < (node a, node b){return a.d > b.d;}
};void dijkstra(int s)
{priority_queue<node> q;memset(dis, INF, sizeof(dis));//dis初始化为INF dis[s] = 0;q.push(node(s, dis[s]));while(!q.empty()){node cur = q.top();q.pop();int from = cur.u;if(cur.d != dis[from])//减少了vis数组,表示该节点被取出来过 continue;for(int i = 0; i < G[from].size(); i++)//更新所有集合外点到集合的dis {Edge e = edges[G[from][i]];if(dis[e.v] > dis[e.u] + e.w){dis[e.v] = dis[e.u] + e.w;pre[e.v] = from;//存储父节点 q.push(node(e.v, dis[e.v]));//将有更新的dis加入到队列中 }}}
}
int main()
{int n, m;while(~scanf("%d%d", &n, &m) && n && m){init(n);for(int i = 0; i < m; i++){int u, v, w;scanf("%d%d%d", &u, &v, &w);AddEdge(u, v, w);AddEdge(v, u, w);}dijkstra(1);printf("%d\n", dis[n]);}return 0;
}

3.SPFA算法

核心思路:

设立一个先进先出的队列用来保存待优化的结点,优化时每次取出队首结点u,并且用u点当前的最短路径估计值对离开u点所指向的结点v进行松弛操作,如果v点的最短路径估计值有所调整,且v点不在当前的队列中,就将v点放入队尾。这样不断从队列中取出结点来进行松弛操作,直至队列空为止
适用情况:可以正权可以有负权,有向图无向图,从单个源点出发到所有结点的最短路

代码参考:
https://blog.csdn.net/Since_natural_ran/article/details/52955460

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
#include<vector>#define inf 0x3f3f3f3fusing namespace std;int dis[105],visit[105];
int n,m;class Node
{
public:int e,v;Node(int a,int b){e = a,v = b;}
};vector<Node>s[105];void spfa()
{memset(dis,inf,sizeof(dis));memset(visit,0,sizeof(visit));dis[1] = 0;queue<int>q;q.push(1);visit[1] = true;while(!q.empty()){int u = q.front();q.pop();visit[u] = false;int num = s[u].size();for(int i = 0;i < num; i++){if(dis[u] + s[u][i].v > dis[s[u][i].e])continue;dis[s[u][i].e] = dis[u] + s[u][i].v;if(!visit[s[u][i].e]){q.push(s[u][i].e);visit[s[u][i].e] = true;}}}
}int main()
{//   freopen("in.txt","r",stdin);while(cin>>n>>m){if(n == 0 && m == 0)break;for(int i = 1;i <= n; i++)s[i].clear();int a,b,c;for(int i = 1;i <= m; i++){cin>>a>>b>>c;s[a].push_back(Node(b,c));s[b].push_back(Node(a,c));//这里无向}spfa();cout<<dis[n]<<endl;}return 0;
}

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

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

相关文章

tomcat java垃圾回收_tomcat启动参数配置,内存和垃圾回收

一般情况下&#xff1a;JAVA_OPTS-Xms2048m -Xmx2048m -XX:MaxPermSize512m -XX:UseParallelGC -XX:ParallelGCThreads8 -XX:UseParallelOldGC -Xloggc:../logs/jvm-gc.log -XX:PrintGCDetails -XX:PrintGCTimeStamps适合于实时要求比较高&#xff1a;-Xms2048m -Xmx2048m -Xmn…

(dijkstra算法+多权值)最短路径问题

给你n个点&#xff0c;m条无向边&#xff0c;每条边都有长度d和花费p&#xff0c;给你起点s终点t&#xff0c;要求输出起点到终点的最短距离及其花费&#xff0c;如果最短距离有多条路线&#xff0c;则输出花费最少的。 Input 输入n,m&#xff0c;点的编号是1~n,然后是m行&am…

java 重定向关键字_SpringMVC 转发、重定向

转发、重定向到其它业务方法org.springframework.stereotype.ControllerRequestMapping("/userController")public classUserController{RequestMapping("/handler1")public String handler1() throwsIOException {//转发给handler2处理return "forwa…

(多源转化成单源dijsktra)一个人的旅行

虽然草儿是个路痴&#xff08;就是在杭电待了一年多&#xff0c;居然还会在校园里迷路的人&#xff0c;汗~),但是草儿仍然很喜欢旅行&#xff0c;因为在旅途中 会遇见很多人&#xff08;白马王子&#xff0c;^0^&#xff09;&#xff0c;很多事&#xff0c;还能丰富自己的阅历&…

最短路模板题

题目&#xff1a; 在每年的校赛里&#xff0c;所有进入决赛的同学都会获得一件很漂亮的t-shirt。但是每当我们的工作人员把上百件的衣服从商店运回到赛场的时候&#xff0c;却是非常累的&#xff01;所以现在他们想要寻找最短的从商店到赛场的路线&#xff0c;你可以帮助他们吗…

php 取经纬度,php根据地址获取百度地图经纬度的实例方法

首先我们来看全部实例代码&#xff1a;/*** param string $address 地址* param string $city 城市名* return array*/function getLatLng($address‘‘,$city‘‘){$result array();$ak ‘‘;//您的百度地图ak&#xff0c;可以去百度开发者中心去免费申请$url "http://…

(kruskal算法复习+模板)Eddy's picture

题目&#xff1a; Eddy begins to like painting pictures recently ,he is sure of himself to become a painter.Every day Eddy draws pictures in his small room, and he usually puts out his newest pictures to let his friends appreciate. but the result it can be…

php解密 eval( base64_decode,PHP之eval(gzinflate(base64_decode加密解密

从网上下载了个php版本的小游戏站的源码&#xff0c;一直都没有时间看&#xff0c;今天闲着没事看了下各个文件夹的内容&#xff0c;有一个文件引起了小胡的注意&#xff0c;文件名为php.php&#xff0c;打开之后发现了是加密过后的php文件&#xff0c;于是试着找了下相关的解密…

(kruskal)Jungle Roads

题目 The Head Elder of the tropical island of Lagrishan has a problem. A burst of foreign aid money was spent on extra roads between villages some years ago. But the jungle overtakes roads relentlessly, so the large road network is too expensive to mainta…

php 留言板分页显示,php有分页的留言板,留言成功后怎么返回当前页?

比如我在【index.php?p3】发布留言&#xff0c;成功后怎么返回到 index.php?p3 这个页面&#xff1f;在【index.php?p5】发布留言成功后怎么返回index.php?p5这个页面&#xff1f;location.href"这里该怎么写&#xff1f;"涉及的页面有index.php&#xff0c;doac…

(最短路)HDU Today(hdu2112)

题目 Problem Description 经过锦囊相助&#xff0c;海东集团终于度过了危机&#xff0c;从此&#xff0c;HDU的发展就一直顺风顺水&#xff0c;到了2050年&#xff0c;集团已经相当规模了&#xff0c;据说进入了钱江肉丝经济开发区500强。这时候&#xff0c;XHD夫妇也退居了…

php文件读取文件内容,PHP文件系统函数-读取文件内容几种方式

介绍几种php获取文件内容的方式介绍读取文件的方式之前&#xff0c;我们先看一下打开文件资源和关闭资源名字资源绑定到一个流 - fopen关闭一个已打开的文件指针 - fclose$handle1 fopen("/home/rasmus/file.txt", "r");fclose($hanle1);$handle2 fopen(…

欧几里得算法和扩展欧几里得算法详解

欧几里得算法&#xff1a; int gcd(int x,int y){if(y) return gcd(y,x%y);return x; }扩展欧几里得算法&#xff1a; 先说一个整体思路&#xff1a; 先求AxBygcd(A,B);的一个解x&#xff0c;y 然后我们可以求他的通解 然后求AxByC的通解 我们先看看怎么求AxBygcd(A,B);的一…

php class使用方法,PHP调试类Krumo使用教程

写程序最讨厌的是程序发生错误&#xff0c;但是却又不知道该从何debug起&#xff0c;我们通常会使用print_r 或者 var_dump 或者是 echo 慢慢的debug。如果你跟我一样使用PHP 5开发&#xff0c;现在可以使用Krumo这个简单好用的工具帮助我们做这件事情。虽然IDE也有内建的debug…

(扩展欧几里得)青蛙的约会

题意 两只青蛙在网上相识了&#xff0c;它们聊得很开心&#xff0c;于是觉得很有必要见一面。它们很高兴地发现它们住在同一条纬度线上&#xff0c;于是它们约定各自朝西跳&#xff0c;直到碰面为止。可是它们出发之前忘记了一件很重要的事情&#xff0c;既没有问清楚对方的特…

(待定系数法)A/B

题目&#xff1a; 要求(A/B)%9973&#xff0c;但由于A很大&#xff0c;我们只给出n(nA%9973)(我们给定的A必能被B整除&#xff0c;且gcd(B,9973) 1)。 Input 数据的第一行是一个T&#xff0c;表示有T组数据。 每组数据有两个数n(0 < n < 9973)和B(1 < B < 10^…

matlab 连接数组,matlab数组操作知识点总结

其实如果单从建模来讲&#xff0c;以下大部分函数都用不到&#xff0c;但是这些都是基础。第一点&#xff1a;数组与矩阵概念的区分数组&#xff1a;与其它编程语言一样&#xff0c;定义是&#xff1a;相同数据类型元素的集合。矩阵&#xff1a;在数学中&#xff0c;矩阵(Matri…

(组合数求模=乘法逆元+快速幂) Problem Makes Problem

题目&#xff1a; As I am fond of making easier problems, I discovered a problem. Actually, the problem is ‘how can you make n by adding k non-negative integers?’ I think a small example will make things clear. Suppose n4 and k3. There are 15 solutions.…

php做图书网站,基于PHP的图书馆网站管理系统的设计与实现

《现代图书情报技术 》 年 第 期 工作交流 总第 期基于 的图书馆网站管理系统的设计与实现 郑婷婷 张 羽 辽宁大学图书馆 沈阳 【摘要 】 阐述了在 下 , 使用 十 设计并实现 了图书馆网站管理系统 。 分系统平台如何搭建 、 系统结构如何设计以及最终如何实现三大部分 。 【关键…

(小费马定理降幂)Sum

题目&#xff1a; 分析与解答&#xff1a; 参考思路&#xff1a; https://www.cnblogs.com/stepping/p/7144512.html https://blog.csdn.net/strangedbly/article/details/50996908 根据隔板定理&#xff0c;把N分成一份的分法数为C(1,n-1)&#xff0c; 把N分成两份的分法…