最短路 dijkstra and floyd
二:最短路算法分析报告
背景
最短路问题(short-path problem):若网络中的每条边都有一个数值(长度、成本、时间等),则找出两节点(通常是源节点和阱节点)之间总权和最小的路径就是最短路问题。最短路问题是网络理论解决的典型问题之一,可用来解决管路铺设、线路安装、厂区布局和设备更新等实际问题。
单源最短路径
包括确定起点的最短路径问题,确定终点的最短路径问题(与确定起点的问题相反,该问题是已知终结结点,求最短路径的问题。在无向图中该问题与确定起点的问题完全等同,在有向图中该问题等同于把所有路径方向反转的确定起点的问题。) 。
求解单源最短路径问题可以采用Dijkstra算法,时间复杂度为O(|V|^2)。Dijkstra算法可以使用斐波那契堆、配对堆等支持Decrease-Key操作的数据结构来进一步优化,优化后的时间复杂度为O(|E|+|V|log|V|)。
Dijkstra只可求无负权的最短路径,因为其目光短浅,看不到后面可以消减的量。在正数中容易得证,若a<b,Dijkstra会取a,若再有一条路c,a+c<b+c是正确的。但引入负数后,可能会出现以下情况

,Dijkstra会先选择A-B这条边,此时A->B的距离固定为1,不再改变,但其实A->B最短路是-1,虽然A->C的最短路是正确的,为-2,但这样的算法是不可使用的。
如果图中有负权回路,可以采用Bellman-Ford算法,算法复杂度是O(|V||E|)。但Bellman-ford算法浪费了许多时间做无必要的松弛,可用SPFA算法进行优化,SPFA算法是用队列进行的优化,优化后时间复杂度为O(k|E|), 其中k为所有顶点进队的平均次数,可以证明k一般小于等于2,由此可见该优化的效果十分显著。
全局最短路径
求图中所有的最短路径可以采用Floyd-Warshall算法,算法时间复杂度为O(|V|^3)。对于稀疏图,还可采用Johnson算法,其采用Bellman-ford和Dijkstra作为其子函数,时间复杂度为O(VElgV)。二者都可计算含负权路径的图,但不可含有负环。
两点最短路径
即已知起点和终点,求两结点之间的最短路径。通常可以用广度优先搜索(BFS)、深度优先搜索(DFS)等方式来实现,时间复杂度是O(|V|)。
本篇报告分析dijkstra算法和floyd算法。
问题描述
在每年的校赛里,所有进入决赛的同学都会获得一件很漂亮的t-shirt。但是每当我们的工作人员把上百件的衣服从商店运回到赛场的时候,却是非常累的!所以现在他们想要寻找最短的从商店到赛场的路线,你可以帮助他们吗?
基本要求
输入包括多组数据。每组数据第一行是两个整数N、M(N<=100,M<=10000),N表示成都的大街上有几个路口,标号为1的路口是商店所在地,标号为N的路口是赛场所在地,M则表示在成都有几条路。N=M=0表示输入结束。接下来M行,每行包括3个整数A,B,C(1<=A,B<=N,1<=C<=1000),表示在路口A与路口B之间有一条路,我们的工作人员需要C分钟的时间走过这条路。输入保证至少存在1条商店到赛场的路线。
测试数据
输入
6 10
1 2 2
1 3 5
1 4 1
2 3 3
2 4 2
3 4 3
3 5 1
3 6 5
4 5 1
5 6 2
0 0
输出
4
算法思想
一:dijkstra
设G=(V,E)是一个带权有向图,把图中顶点集合V分成两组,第一组为已求出最短路径的顶点集合(用S表示,初始时S中只有一个源点,以后每求得一条最短路径 , 就将 加入到集合S中,直到全部顶点都加入到S中,算法就结束了),第二组为其余未确定最短路径的顶点集合(用U表示),按最短路径长度的递增次序依次把第二组的顶点加入S中。
在加入的过程中,总保持从源点v到S中各顶点的最短路径长度不大于从源点v到U中任何顶点的最短路径长度。此外,每个顶点对应一个距离,S中的顶点的距离就是从v到此顶点的最短路径长度,U中的顶点的距离是从v到此顶点只包括S中的顶点为中间顶点的当前最短路径长度。
二:floyd
(1)初始时设置一个n阶方阵,令其对角线元素为0,若存在弧<Vi,Vj>,则对应元素为权值,否则为。
(2)逐步试着在原直接路径中增加中间顶点,若加入中间点后路径变短,则修改之;否则,维持原值。
(3)所有顶点试探完毕,算法结束。
实现过程
一:dijkstra
|
终点 |
从V1到各终点的最短路径及长度 |
|||||
|
V2 |
2 <V1,V2> |
2 <V1,V2> |
------- |
------ |
-------- |
------- |
|
V3 |
5 <V1,V3> |
4 <V1,V4,V3> |
4 <V1,V4,V3> |
3 <V1,V4,V5,V3> |
------- |
--------- |
|
V4 |
1 <V1,V4> |
---------- |
------- |
-------- |
------- |
--------- |
|
V5 |
INF |
2 <V1,V4,V5> |
2 <V1,V4,V5> |
--------- |
------ |
-------- |
|
V6 |
INF |
INF |
INF |
4 <V1,V4,V5,V6> |
4 <V1,V4,V5,V6> |
-------- |
|
|
V4:1 <V1,V4> |
V2:2 <V1,V2> |
V5:2 <V1,V4,V5> |
V3:3 <V1,V4,V5,V3> |
V6:4 <V1,V4,V5,V6> |
|
图用带权邻接矩阵存储map[][]
数组dis[]存放当前找到的从源点V0到每个终点的最短路径长度,其初态为图中直接路径权值。
数组vis[]表示从是否访问过此点。
二:floyd
逐步试着在原直接路径中增加中间顶点,若加入中间点后路径变短,则修改之;否则,维持原值。
初始
|
|
1 |
2 |
3 |
4 |
5 |
6 |
|
1 |
0 |
2 |
5 |
1 |
INF |
INF |
|
2 |
2 |
0 |
3 |
2 |
INF |
INF |
|
3 |
5 |
3 |
0 |
3 |
1 |
5 |
|
4 |
1 |
2 |
3 |
0 |
1 |
INF |
|
5 |
INF |
INF |
1 |
1 |
0 |
2 |
|
6 |
INF |
INF |
5 |
INF |
2 |
0 |
加入1
|
|
1 |
2 |
3 |
4 |
5 |
6 |
|
1 |
0 |
2 |
5 |
1 |
INF |
INF |
|
2 |
|
0 |
3 |
2 |
INF |
INF |
|
3 |
|
|
0 |
3 |
1 |
5 |
|
4 |
|
|
|
0 |
1 |
INF |
|
5 |
|
|
|
|
0 |
2 |
|
6 |
|
|
|
|
|
0 |
加入2
|
|
1 |
2 |
3 |
4 |
5 |
6 |
|
1 |
0 |
2 |
5 |
1 |
INF |
INF |
|
2 |
|
0 |
5 |
2 |
INF |
INF |
|
3 |
|
|
0 |
3 |
1 |
5 |
|
4 |
|
|
|
0 |
1 |
INF |
|
5 |
|
|
|
|
0 |
2 |
|
6 |
|
|
|
|
|
0 |
加入3
|
|
1 |
2 |
3 |
4 |
5 |
6 |
|
1 |
0 |
2 |
5 |
1 |
INF |
10 |
|
2 |
|
0 |
5 |
2 |
4 |
8 |
|
3 |
|
|
0 |
3 |
1 |
5 |
|
4 |
|
|
|
0 |
1 |
8 |
|
5 |
|
|
|
|
0 |
2 |
|
6 |
|
|
|
|
|
0 |
加入4
|
|
1 |
2 |
3 |
4 |
5 |
6 |
|
1 |
0 |
2 |
4 |
1 |
2 |
9 |
|
2 |
|
0 |
3 |
2 |
3 |
8 |
|
3 |
|
|
0 |
3 |
1 |
5 |
|
4 |
|
|
|
0 |
1 |
8 |
|
5 |
|
|
|
|
0 |
2 |
|
6 |
|
|
|
|
|
0 |
加入5
|
|
1 |
2 |
3 |
4 |
5 |
6 |
|
1 |
0 |
2 |
3 |
1 |
2 |
4 |
|
2 |
|
0 |
3 |
2 |
3 |
5 |
|
3 |
|
|
0 |
2 |
1 |
3 |
|
4 |
|
|
|
0 |
1 |
3 |
|
5 |
|
|
|
|
0 |
2 |
|
6 |
|
|
|
|
|
0 |
代码实现
一:dijkstra
#include<stdio.h>
#include<string.h>
#include<iostream>
#include<algorithm>
using namespace std;
#define INF 0xfffffff
int pri[][];//两个顶点之间距离
int dis[];//起点到该点的最短距离
int vis[];//标记数组
int n,m; void dijkstra()
{
memset(vis,,sizeof(vis));
vis[]=;
for(int i=;i<=n;i++)
dis[i]=pri[][i];
for(int i=;i<n;i++)
{
int M=INF,k=-;
//每次找出最小的距离加入到集合
for(int j=;j<=n;j++)
{
if(!vis[j]&&dis[j]<M)
M=dis[j],k=j;
}
if(k==-)
return ;
vis[k]=;//加入集合
//新加入一个顶点,更新到达各个顶点的距离
for(int j=;j<=n;j++)
if(!vis[j]&&dis[j]>dis[k]+pri[k][j])
dis[j]=dis[k]+pri[k][j];
}
}
int main()
{
while(scanf("%d%d",&n,&m)!=EOF,n||m)
{
//初始化
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
pri[i][j]=i==j?:INF;
for(int i=;i<=m;i++)
{
int a,b,c;scanf("%d%d%d",&a,&b,&c);
if(pri[a][b]>c)//防止重边
pri[a][b]=pri[b][a]=c;//这个是无向图的存储
}
dijkstra();
printf("%d\n",dis[n]);
}
return ;
}
二:floyd
#include<stdio.h>
#include<string.h>
#include<iostream>
#include<algorithm>
using namespace std;
#define INF 0xfffffff
int pri[][];//两个顶点之间距离
int dis[];//起点到该点的最短距离
int vis[];//标记数组
int n,m;
void floyd()
{
for(int k=;k<=n;k++)//中间点
{
for(int i=;i<=n;i++)
{
for(int j=;j<=n;j++)
{
pri[i][j]=min(pri[i][j],pri[i][k]+pri[k][j]);//取当前最短距离和含有中间顶点的距离的最小值
}
}
}
}
int main()
{
while(scanf("%d%d",&n,&m)!=EOF,n||m)
{
//初始化
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
pri[i][j]=i==j?:INF;
for(int i=;i<=m;i++)
{
int a,b,c;scanf("%d%d%d",&a,&b,&c);
if(pri[a][b]>c)//防止重边
pri[a][b]=pri[b][a]=c;//这个是无向图的存储
}
floyd();
int a,b;
scanf("%d%d",&a,&b);
printf("%d\n",pri[a][b]);
}
return ;
}
运行截图
一:dijkstra

二:floyd

个人总结
Dijkstra算法的时间复杂度为O(n^2) ,较之Floyd算法有很大提升,但是由于使用的是邻接矩阵的存储,所以说当顶点数过大的时候,我们就不可能用二维数组来存储了。切记每次加入一个点时要更新最短距离。Floyd算法的时间复杂度为(n^3),虽然代码简单但当n很大时会超时。
最短路 dijkstra and floyd的更多相关文章
- 算法学习笔记(三) 最短路 Dijkstra 和 Floyd 算法
图论中一个经典问题就是求最短路.最为基础和最为经典的算法莫过于 Dijkstra 和 Floyd 算法,一个是贪心算法,一个是动态规划.这也是算法中的两大经典代表.用一个简单图在纸上一步一步演算,也是 ...
- 最短路——Dijkstra和Floyd
Problem Description 在每年的校赛里,所有进入决赛的同学都会获得一件很漂亮的t-shirt.但是每当我们的工作人员把上百件的衣服从商店运回到赛场的时候,却是非常累的!所以现在他们想要 ...
- 最短路(Dijkstra,Floyd,Bellman_Ford,SPFA)
当然,这篇文章是借鉴大佬的... 最短路算法大约来说就是有4种——Dijkstra,Floyd,Bellman_Ford,SPFA 接下来,就可以一一看一下... 1.Dijkstra(权值非负,适用 ...
- 训练指南 UVALive - 4080(最短路Dijkstra + 边修改 + 最短路树)
layout: post title: 训练指南 UVALive - 4080(最短路Dijkstra + 边修改 + 最短路树) author: "luowentaoaa" ca ...
- hdu-2544-最短路(dijkstra算法模板)
题目链接 题意很清晰,入门级题目,适合各种模板,可用dijkstra, floyd, Bellman-ford, spfa Dijkstra链接 Floyd链接 Bellman-Ford链接 SPFA ...
- (最短路径算法整理)dijkstra、floyd、bellman-ford、spfa算法模板的整理与介绍
这一篇博客以一些OJ上的题目为载体.整理一下最短路径算法.会陆续的更新... 一.多源最短路算法--floyd算法 floyd算法主要用于求随意两点间的最短路径.也成最短最短路径问题. 核心代码: / ...
- hdu 2544 最短路 Dijkstra
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2544 题目分析:比较简单的最短路算法应用.题目告知起点与终点的位置,以及各路口之间路径到达所需的时间, ...
- 单源最短路dijkstra算法&&优化史
一下午都在学最短路dijkstra算法,总算是优化到了我能达到的水平的最快水准,然后列举一下我的优化历史,顺便总结总结 最朴素算法: 邻接矩阵存边+贪心||dp思想,几乎纯暴力,luoguTLE+ML ...
- HUD.2544 最短路 (Dijkstra)
HUD.2544 最短路 (Dijkstra) 题意分析 1表示起点,n表示起点(或者颠倒过来也可以) 建立无向图 从n或者1跑dij即可. 代码总览 #include <bits/stdc++ ...
随机推荐
- (Map)利用Map,完成下面的功能: 从命令行读入一个字符串,表示一个年份,输出该年的世界杯冠军是哪支球队。如果该 年没有举办世界杯,则输出:没有举办世界杯。 附:世界杯冠军以及对应的夺冠年份,请参考本章附录。 附录 (Map)在原有世界杯Map 的基础上,增加如下功能: 读入一支球队的名字,输出该球队夺冠的年份列表。 例如,读入“巴西”,应当输出 1958 1962 1970 1
package homework001; import java.util.HashMap; import java.util.Scanner; public class Map { public s ...
- Volley使用指南第一回(来自developer.android)
最近闲来想看看android网络方面的东西.google在2013年发布了一个叫做Volley的网络请求框架,我看了一下官网,居然在training里面就有教程.首先,英文的东西看着 还是挺不爽的,特 ...
- linux tomcat自启动设置
国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...
- webViewDidFinishLoad因为网页里的重定向,会调用多次,使用web view.isLoading来解决
我编码如下,但我发现 webViewDidFinishLoad() 会发生若干次. 如何知道 webViewDidFinishLoad() 最后发生吗? iNavigate = ; - (void)w ...
- Flash Builder 创建CSS
1.global 选择器将样式应用于所有控件 在 Flash Builder 中创建新MXML 文件并切换到设计模式 属性视图右侧的外观视图可更改外观 Flash Builder 自动创建CS ...
- hdu 5480 Conturbatio 线段树 单点更新,区间查询最小值
Conturbatio Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=54 ...
- 初学Node.js第一天
最近开始下班到家不知道该做啥,因为水平太菜,要学的东西实在太多,反而陷入了不知道该学什么的困境,结果天天就是看别人的博客,看到什么标题比较感兴趣就点进去,没有一个目标. 今天突然兴起,决定要捣鼓捣鼓N ...
- EntityFramework小知识
Entity Framework 应用程序有以下优点: 1 应用程序可以通过更加以应用程序为中心的概念性模型(包括具有继承性.复杂成员和关系的类型)来工作. 2 应用程序不再对特定的数据引擎或存储架构 ...
- 第一章 01 namespace 命名空间
一.什么是namespace? namesapce是为了防止名字冲突提供的一种控制方式. 当一个程序需要用到很多的库文件的时候,名字冲突有时无法避免.之前的解决思路是使用更长的变量名字,使用不方便. ...
- PS基础
1.仿制图章工具:[小 ]大 建立一个新的图层,可以进行图片某个部分的复制,完全复制之后,还可以调整大小(ctrl+t), 颜色(ctrl+u打开色相饱和度的菜单)等. 2.修复画笔工具:与仿制 ...