图论——最短路径 Dijkstra算法、Floyd算法
1.弗洛伊德算法(Floyd)
弗洛伊算法核心就是三重循环,M [ j ] [ k ] 表示从 j 到 k 的路径,而 i 表示当前 j 到 k 可以借助的点;红色部分表示,如果 j 到 i ,i 到 k 是通的,就将 j 到 k 的值更新为
M[j][i] + M[i][k] 和 M[j][k] 较短的一个。
<<; ; i <= n; i++) { ; j <= n; j++) { ; k <= n; k++) { if (j!=k) { M[j][k] = min(M[j][i] + M[i][k] , M[j][k]); } } } }
给个题目链接,写完可以交试一下:http://www.dotcpp.com/oj/problem1709.html
完整代码:
#include <iostream> #include <queue> using namespace std; #define inf 2147483647 ][]; int main() { int n; queue<int>q; cin >> n; ; i <= n; i++) { ; j <= n; j++) { cin >> M[i][j]; && i != j)M[i][j] = inf; } } ; i <= n; i++) { ; j <= n; j++) { ; k <= n; k++) { ) { if (M[j][i] != inf && M[i][k] != inf) { M[j][k] = M[j][i] + M[i][k] < M[j][k] ? M[j][i] + M[i][k] : M[j][k]; } } } } } ; i <= n; i++) { ; j <= n; j++) { << " "; else cout << M[i][j] << " "; } cout << endl; } ; }
2.迪杰斯特拉
Floyd只要暴力的三个for就可以出来,代码很好理解,但缺点就是时间复杂度高是O(n³)。Dijkstra的时间复杂度是O(n²),要快很多。
不过要注意这个算法所求的是单源最短路。所以说,如果题目是求任意一对顶点间的最短路径问题,那就需要对每个顶点进行一遍迪杰斯特拉算法,这种情况就适合弗洛伊德算法了。
思想图解:
用dis数组实时记录起始点(起始点取1) 到达的所有节点的距离。(自己到自己的路径长度 0,到不了的点是 inf(极大值))
dis数组初始值是这样的,4是当前距离节点1最近的点。(已经访问过的,我们标记上不再次访问)
借助4节点,对dis数组进行更新(如果有更短的路径,就对dis数组进行值替换),走到2,无操作。
借助3节点,对dis数组进行更新,最后走到5节点,退出。(实际过程中,走到最后一个节点,别的节点都访问过,进行标记了,什么也不会做)。
借助3节点,对dis数组进行更新
测试题目:http://acm.hdu.edu.cn/showproblem.php?pid=2544 (数据很弱,AC了,实现也不一定是正确的,强烈建议再做后面一题)
#include <iostream> #include <algorithm> #include <string.h> using namespace std; << ; int n, m; ]; ][]; ]; void initialize() { memset(book, , sizeof(book)); ; i <= n; i++) { ; j <= n; j++) { if (i != j)M[i][j] = inf; } } } void dijkstra() { while (true) { ; ; i <= n; i++) { || dis[i] < dis[v])) v = i;//从dis数组中找出当前距离起点最短的节点 } ) break; book[v] = true; ; i <= n; i++) { dis[i] = min(dis[i], dis[v] + M[v][i]); } } } int main() { while (cin >> n >> m) { && m == )break; initialize(); ; i < m; i++) { int A, B, C; cin >> A >> B >> C; M[A][B] = C; M[B][A] = C; } book[] = true; ; i <= n; i++) { dis[i] = M[][i]; } dijkstra(); cout << dis[n] << endl; } ; }
使用优先级队列优化查找过程(理论上是要更快的,但是我交上去,时间反而更慢了):
#include <iostream> #include <queue> #include <algorithm> using namespace std; << ; int n, m; ][]; ]; class Node { public: int to,distance; Node(int t, int d) { to = t; distance = d; } bool operator< (Node a) const{ return a.distance < distance; } }; priority_queue<Node>q; void initialize() { ; i <= n; i++) { ; j <= n; j++) { if (i != j)M[i][j] = inf; } } } void dijkstra() { while (!q.empty()) { int min; min = q.top().to;q.pop(); //处理掉之前push进去的,距离较长的边,不写不会错,效率会降低。 while (!q.empty() && q.top().to == min) { q.pop(); } ; i <= n; i++) { if (dis[i] > dis[min] + M[min][i]) { dis[i] = dis[min] + M[min][i]; q.push(Node(i, dis[i])); } } } } int main() { while (cin >> n >> m) { && m == )break; initialize(); ; i < m; i++) { int A, B, C; cin >> A >> B >> C; M[A][B] = C; M[B][A] = C; } ; i <= n; i++) { dis[i] = M[][i]; && dis[i] != inf) { q.push(Node(i, dis[i])); } } dijkstra(); cout << dis[n] << endl; } ; }
邻接表+优先级队列:(邻接表的迪杰斯特拉实现起来,要复杂得多,但是跑起来确实比较快)
#include <iostream> #include <stdio.h> #include <queue> using namespace std; ]; << ; ]; class ENode { public: int to; int dis; ENode *next = NULL; ENode() {}; ENode(int t, int d) { dis = d; to = t; } void push(int t, int d) { ENode *p = new ENode; p->to = t; p->dis = d; p->next = next; next = p; } bool operator<(ENode e)const { return e.dis < dis; } }head[]; int main() { priority_queue<ENode>q; int n, m, c1, c2, c3; while (cin >> n >> m) { && m == )break; ; i <= n; i++) { dis[i] = inf; fuck[i] = false; } ; i < m; i++) { scanf("%d%d%d", &c1, &c2, &c3); head[c1].push(c2, c3); head[c2].push(c1, c3); || c2 == ) { dis[c1 == ? c2 : c1] = c3; q.push(*head[c1].next); } } fuck[] = true; while (!q.empty()) { int fm = q.top().to; q.pop(); if (fuck[fm])continue; fuck[fm] = true; ENode *p = head[fm].next; while (p) { int me = p->to; if (dis[me] > dis[fm] + p->dis) { dis[me] = dis[fm] + p->dis; q.push(ENode(me, dis[me])); } p = p->next; } } printf("%d\n", dis[n]); ; i <= n; i++) { ENode *p = head[i].next; while (p) { ENode *t = p->next; delete p; p = t; } head[i].next = NULL; } } ; }
理解不深刻,实现出来的错误版本,能出来正确的答案,但升高了复杂度:
#include <iostream> #include <stdio.h> #include <queue> using namespace std; ]; ]; << ; class ENode { public: int to; int dis; ENode *next = NULL; void push(int t, int d) { ENode *p = new ENode;; p->to = t; p->dis = d; p->next = next; next = p; } bool operator<(ENode e)const { return e.dis < dis; } }head[]; int main() { priority_queue<ENode>q; int n, m, c1, c2, c3; while (cin >> n >> m) { && m == )break; ; i <= n; i++) { dis[i] = inf; } ; i < m; i++) { scanf("%d%d%d", &c1, &c2, &c3); head[c1].push(c2, c3); head[c2].push(c1, c3); } ENode *p = head[].next; while (p) { q.push(*p); dis[p->to] = p->dis; p = p->next; } while (!q.empty()) { int fm = q.top().to; q.pop(); /*------------*/ /*这个是错误指定背锅店,如果不写这一句,当出现完全图的情况时, 算法会近乎退化成n^n,HDU的这题测试数据太弱了,导致我没有发现问题。 写了的话,由于下面往队列中Push的错误的,这里也就成了背锅点*/ if (fuck[fm])continue; fuck[fm] = true; /*------------*/ ENode *p = head[fm].next; while (p) { int me = p->to; if (dis[me] > dis[fm] + p->dis) { dis[me] = dis[fm] + p->dis; //真正错误在这里 q.push(*p); } p = p->next; } } printf("%d\n", dis[n]); ; i <= n; i++) { ENode *p = head[i].next; while (p) { ENode *t = p->next; delete p; p = t; } head[i].next = NULL; } } ; }
我为什么能发现代码实现的错误,因为我天赋过人,因为这题数据很强,https://www.luogu.org/problemnew/show/P4779 数据量很大,100000个节点,不管是时间还是空间上,不能再用邻接矩阵了;邻接表,如果实现得不恰当也是会超时的,这题如果能AC,那算法实现的肯定是没有问题了。
邻接表+优先级队列:
#include <iostream> #include <stdio.h> #include <queue> using namespace std; ]; ]; << ; class ENode { public: int to; int dis; ENode *next = NULL; void push(int t, int d) { ENode *p = new ENode;; p->to = t; p->dis = d; p->next = next; next = p; } bool operator<(ENode e)const { return e.dis < dis; } }head[]; int main() { priority_queue<ENode>q; int n, m, s, c1, c2, c3; cin >> n >> m >> s; ; i < m; i++) { //cin >> c1 >> c2 >> c3; scanf("%d%d%d", &c1, &c2, &c3); head[c1].push(c2, c3); } ; i <= n; i++) { if (i != s) { dis[i] = inf; } } ENode *p = head[s].next; while (p) { || (p->dis < dis[p->to])) { q.push(*p); dis[p->to] = p->dis; } p = p->next; } while (!q.empty()) { //获得当期距离 源点 最近的点 int min = q.top().to; q.pop(); if (fuck[min])continue; fuck[min] = true; ENode *p = head[min].next; while (p) { int to = p->to; if (dis[to] > dis[min] + p->dis) { dis[to] = dis[min] + p->dis; ENode e = *p; e.dis = dis[to]; q.push(e); } p = p->next; } } ; i <= n; i++) { printf("%d ", dis[i]); } cout << endl; ; }
3、Bellman-Ford算法
Bellman - ford效率较低,代码难度较小。重要的是若给定的图存在负权边,Dijkstra算法便没有了用武之地,Bellman - ford算法便派上用场了。
图论——最短路径 Dijkstra算法、Floyd算法的更多相关文章
- 算法学习笔记(三) 最短路 Dijkstra 和 Floyd 算法
图论中一个经典问题就是求最短路.最为基础和最为经典的算法莫过于 Dijkstra 和 Floyd 算法,一个是贪心算法,一个是动态规划.这也是算法中的两大经典代表.用一个简单图在纸上一步一步演算,也是 ...
- 多源最短路径算法—Floyd算法
前言 在图论中,在寻路最短路径中除了Dijkstra算法以外,还有Floyd算法也是非常经典,然而两种算法还是有区别的,Floyd主要计算多源最短路径. 在单源正权值最短路径,我们会用Dijkstra ...
- 图论-最短路径<Dijkstra,Floyd>
昨天: 图论-概念与记录图的方法 以上是昨天的Blog,有需要者请先阅读完以上再阅读今天的Blog. 可能今天的有点乱,好好理理,认真看完相信你会懂得 分割线 第二天 引子:昨天我们简单讲了讲图的概念 ...
- Dijkstra与Floyd算法
1. Dijkstra算法 1.1 定义概览 Dijkstra(迪杰斯特拉)算法是典型的单源最短路径算法,用于计算一个节点到其他所有节点的最短路径.主要特点是以起始点为中心向外层层扩展,直到扩展到终点 ...
- [链接]最短路径的几种算法[迪杰斯特拉算法][Floyd算法]
最短路径—Dijkstra算法和Floyd算法 http://www.cnblogs.com/biyeymyhjob/archive/2012/07/31/2615833.html Dijkstra算 ...
- JS实现最短路径之弗洛伊德(Floyd)算法
弗洛伊德算法是实现最小生成树的一个很精妙的算法,也是求所有顶点至所有顶点的最短路径问题的不二之选.时间复杂度为O(n3),n为顶点数. 精妙之处在于:一个二重初始化,加一个三重循环权值修正,完成了所有 ...
- 图的最短路径算法-- Floyd算法
Floyd算法求的是图的任意两点之间的最短距离 下面是Floyd算法的代码实现模板: ; ; // maxv为最大顶点数 int n, m; // n 为顶点数,m为边数 int dis[maxv][ ...
- 最短路-SPFA算法&Floyd算法
SPFA算法 算法复杂度 SPFA 算法是 Bellman-Ford算法 的队列优化算法的别称,通常用于求含负权边的单源最短路径,以及判负权环. SPFA一般情况复杂度是O(m)最坏情况下复杂度和朴素 ...
- 只有5行代码的算法——Floyd算法
Floyd算法用于求一个带权有向图(Wighted Directed Graph)的任意两点距离的算法,运用了动态规划的思想,算法的时间复杂度为O(n^3).具体方法是:设点i到点j的距离为d[i][ ...
随机推荐
- Solr环境配置
1.准备 第一步下载JDK1.8.0_131Tomcat9.0.7 注意版本的兼容性 第二歩下载solr,目前使用的是solr-7.3.0 2.安装 1. 将 solr 压缩包解压,并将solr- ...
- 一、cent OS安装配置JDK
到oracle官网下载JDKhttp://www.oracle.com/technetwork/java/javase/downloads/index-jsp-138363.html 在cent OS ...
- Spring MVC 实现Excel的导入导出功能(1:Excel的导入)
简介 这篇文章主要记录自己学习上传和导出Excel时的一些心得,企业办公系统的开发中,经常会收到这样的需求:批量录入数据.数据报表使用 Excel 打开,或者职能部门同事要打印 Excel 文件,而他 ...
- 经典的SQL面试题及答案
- scrum 项目的基本模式
scrum 项目中有 3 个主要的角色:产品所有者, Scrum 主管和团队成员 产品所有者和团队其他成员一起工作,负责维护生产积压工作表 (production backlog) ,并对表中的项制定 ...
- hadoop classpath 的作用
HADOOP_CLASSPATH 是设置要运行的类的路径.否则当你用hadoop classname [args]方式运行程序时会报错,说找不到要运行的类.用hadoop jar jar_name.j ...
- BZOJ1968 [Ahoi2005] 约数研究
Description Input 只有一行一个整数 N(0 < N < 1000000). Output 只有一行输出,为整数M,即f(1)到f(N)的累加和. Sample Input ...
- bootstrap学习笔记细化(按钮)
button:btn 圆角灰色按钮 button:btn btn-default 圆角灰色边框按钮 button:btn btn-success 绿色 button:btn btn-primary 蓝 ...
- MUI框架-05-用MUI做一个简单App
MUI框架-05-用MUI做一个简单App MUI 是一个前端框架,前端框架就像 Bootstrap,EasyUI,Vue ,为了做 app 呢,就有了更加高效的 MUI,我觉得前端框架有很多,也没有 ...
- js API列表
// 主要是ES的API和一小部分浏览器的API. // 新加入标准的API有可能是浏览器事实上早已实现的. // ECMAScript目前是每年都会发布新版本(目前已经相对稳定,每年都会又增 ...