Algorithm --> Dijkstra和Floyd最短路径算法
Dijkstra算法
一.最短路径的最优子结构性质
该性质描述为:如果P(i,j)={Vi....Vk..Vs...Vj}是从顶点i到j的最短路径,k和s是这条路径上的一个中间顶点,那么P(k,s)必定是从k到s的最短路径。下面证明该性质的正确性。
假设P(i,j)={Vi....Vk..Vs...Vj}是从顶点i到j的最短路径,则有P(i,j)=P(i,k)+P(k,s)+P(s,j)。而 P(k,s)不是从k到s的最短距离,那么必定存在另一条从k到s的最短路径P'(k,s),那么 P'(i,j)=P(i,k)+P'(k,s)+P(s,j)<P(i,j)。则与P(i,j)是从i到j的最短路径相矛盾。因此该性质得证。
二.Dijkstra算法
由上述性质可知,如果存在一条从i到j的最短路径(Vi.....Vk,Vj),Vk是Vj前面的一顶点。那么(Vi...Vk)也必定是从i到k的最短 路径。为了求出最短路径,Dijkstra就提出了以最短路径长度递增,逐次生成最短路径的算法。譬如对于源顶点V0,首先选择其直接相邻的顶点中长度最 短的顶点Vi,那么当前已知可得从V0到达Vj顶点的最短距离dist[j]=min{dist[j],dist[i]+matrix[i][j]}。根 据这种思路,
假设存在G=<V,E>,源顶点为V0,U={V0},dist[i]记录V0到i的最短距离,path[i]记录从V0到i路径上的i前面的一个顶点。
1.从V-U中选择使dist[i]值最小的顶点i,将i加入到U中;
2.更新与i直接相邻顶点的dist值。(dist[j]=min{dist[j],dist[i]+matrix[i][j]})
3.知道U=V,停止。
三.算法实例
先给出一个无向图

用Dijkstra算法找出以A为起点的单源最短路径步骤如下

代码:
#include <iostream>
#include <cstring>
using namespace std; #define DEBUG 0
#define INF 0x7fffffff
#define MAX 20 int N, V;
int graph[MAX][MAX];
int dist[MAX];
int prev[MAX];
bool visited[MAX]; void PrintPath()
{
for( int i = ; i <= V; i++ )
{
cout << "1 --> " << i << " : " << dist[i] << endl;
} for( int i = ; i <= V; i++ ) //计算从1到每个顶点的距离
{
int path[MAX] = {};
int step = ;
int cur = i;
do
{
path[step++] = cur;
cur = prev[cur];
}
while( cur != - ); //前一个点为-1,则结束 for( int j = step - ; j >= ; j-- )
{
cout << path[j] << " ";
}
cout << endl;
}
} int GetMinDist()
{
int index, min = INF;
for( int i = ; i <= V; i++ )
{
if( !visited[i] && min > dist[i] )
{
min = dist[i];
index = i;
}
}
return index;
} void Dijkstra( int v )
{
for( int i = ; i <= V; i++ )
{
if( graph[v][i] == INF )
{
dist[i] = INF;
prev[i] = -;
}
else
{
dist[i] = graph[v][i];
prev[i] = v;
}
visited[i] = false;
} dist[v] = ;
visited[v] = true; for( int i = ; i < V; i++ ) //迭代V-1次,不用计算源点了,还剩下V-1个需要计算的顶点
{
int u = GetMinDist(); visited[u] = true; for( int j = ; j <= V; j++ ) //更新dist数组
{
if( visited[j] == false && graph[u][j] != INF && dist[u] != INF && dist[j] > dist[u] + graph[u][j] )
{
dist[j] = dist[u] + graph[u][j];
prev[j] = u;
}
}
}
} void InitData()
{
memset( visited, false, sizeof( visited ) ); //初始化visited for( int i = ; i <= V; i++ )
{
for( int j = ; j <= V; j++ )
{
graph[i][j] = INF;
}
dist[i] = INF;
}
} int main()
{
int a, b, value;
while( cin >> V, V ) // 输入顶点数
{
cin >> N; //输入边数
InitData();
for( int i = ; i <= N; i++ )
{
cin >> a >> b >> value;
graph[a][b] = graph[b][a] = value;
} Dijkstra( );
PrintPath();
}
}
输入文件:
/* 6 10
1 2 4
1 3 8
2 3 3
2 4 4
2 5 6
3 4 2
3 5 2
4 5 4
4 6 9
5 6 4 result :
1 --> 1 : 0
1 --> 2 : 4
1 --> 3 : 7
1 --> 4 : 8
1 --> 5 : 9
1 --> 6 : 13
1
1 2
1 2 3
1 2 4
1 2 3 5
1 2 3 5 6 */
Floyd算法
1.算法思想原理:
Floyd算法是一个经典的动态规划算法。从任意节点i到任意节点j的最短路径不外乎2种可能,1是直接从i到j,2是从i经过若干个节点k到j。所以,假设Dis(i,j)为节点u到节点v的最短路径的距离,对于每一个节点k,检查Dis(i,k) + Dis(k,j) < Dis(i,j)是否成立,如果成立,证明从i到k再到j的路径比i直接到j的路径短,则设置Dis(i,j) = Dis(i,k) + Dis(k,j),这样一来,当遍历完所有节点k,Dis(i,j)中记录的便是i到j的最短路径的距离。
2.算法描述:
a.从任意一条单边路径开始。所有两点之间的距离是边的权,如果两点之间没有边相连,则权为无穷大。
b.对于每一对顶点 u 和 v,看看是否存在一个顶点 w 使得从 u 到 w 再到 v 比己知的路径更短。如果是更新它。
3.Floyd算法过程矩阵的计算----十字交叉法
方法:两条线,从左上角开始计算一直到右下角 如下所示
给出矩阵,其中矩阵A是邻接矩阵,而矩阵Path记录u,v两点之间最短路径所必须经过的点

相应计算方法如下:



最后A3即为所求结果.
算法实现:
#include <iostream> using namespace std; #define INF 65536
#define MAX 20 int graph[MAX][MAX];
int KeyPoint[MAX][MAX];
int V, E; void PrintPath()
{
cout << graph[][V] << endl; int path[MAX];
int step = ;
int cur = V; while(cur != ) {
path[step++] = cur;
cur = KeyPoint[][cur];
} path[step++] = ; //保存起点 for (int j = step - ; j >= ; j--)
{
cout << path[j] << " ";
}
cout << endl;
} void Floyd()
{
graph[][] = ; for(int k = ; k <= V; k++) //对于每一个节点k,检查Dis(i,k) + Dis(k,j) < Dis(i,j)是否成立
for(int i = ; i <= V; i++)
for(int j = ; j <= V; j++)
if(graph[i][j] > graph[i][k] + graph[k][j])
{
graph[i][j] = graph[i][k] + graph[k][j];
KeyPoint[i][j] = k;
}
PrintPath();
} void InitData()
{
for(int i = ; i <= V; i++)
{
for(int j = ; j <= V; j++)
{
graph[i][j] = INF;
KeyPoint[i][j] = ;
}
}
} int main()
{
int a, b, length;
while(cin >> V, V) //输入顶点数
{
InitData(); cin >> E; //输入边数
for(int i = ; i <= E; i++)
{
cin >> a >> b >> length;
graph[a][b] = graph[b][a] = length;
} Floyd();
}
}
测试用例
/* 6 10
1 2 4
1 3 8
2 3 3
2 4 4
2 5 6
3 4 2
3 5 2
4 5 4
4 6 9
5 6 4 result:
13
1 2 3 5 6 */
Algorithm --> Dijkstra和Floyd最短路径算法的更多相关文章
- Dijkstra 单源最短路径算法
Dijkstra 算法是一种用于计算带权有向图中单源最短路径(SSSP:Single-Source Shortest Path)的算法,由计算机科学家 Edsger Dijkstra 于 1956 年 ...
- Floyd最短路径算法
看完这篇文章写的小程序,Floyd最短路径算法,求从一个点到另一个点的最短距离,中间可以经过其他任意个点.三个for循环,从i到j依次经过k的最短距离,最外层for循环是经过点K,内部两个循环是从i( ...
- Dijkstra单源点最短路径算法
学习参考: Dijkstra算法(单源最短路径) 最短路径—Dijkstra算法和Floyd算法 使用的图结构: 邻接矩阵: -1 20 -1 25 80-1 -1 40 -1 -1-1 -1 -1 ...
- Floyd最短路径算法(来自微信公众号“算法爱好者”改编)
暑假,小哼准备去一些城市旅游.有些城市之间有公路,有些城市之间则没有,如下图.为了节省经费以及方便计划旅程,小哼希望在出发之前知道任意两个城市之前的最短路程. 上图中有4个城市8条公路,公路上的数字表 ...
- 【模板 && 拓扑】 Dijkstra 单源最短路径算法
话不多说上代码 链式前向星233 #include<bits/stdc++.h> using namespace std; ,_max=0x3fffffff; //链式前向星 struct ...
- 图之单源Dijkstra算法、带负权值最短路径算法
1.图类基本组成 存储在邻接表中的基本项 /** * Represents an edge in the graph * */ class Edge implements Comparable< ...
- 一篇文章讲透Dijkstra最短路径算法
Dijkstra是典型最短路径算法,计算一个起始节点到路径中其他所有节点的最短路径的算法和思想.在一些专业课程中如数据结构,图论,运筹学等都有介绍.其思想是一种基础的求最短路径的算法,通过基础思想的变 ...
- Johnson 全源最短路径算法
解决单源最短路径问题(Single Source Shortest Paths Problem)的算法包括: Dijkstra 单源最短路径算法:时间复杂度为 O(E + VlogV),要求权值非负: ...
- Floyd-Warshall 全源最短路径算法
Floyd-Warshall 算法采用动态规划方案来解决在一个有向图 G = (V, E) 上每对顶点间的最短路径问题,即全源最短路径问题(All-Pairs Shortest Paths Probl ...
随机推荐
- windows 7蓝屏解决办法
1.通过工具打开出现蓝屏原因的dmp文件,找到原因 Unable to load image \SystemRoot\system32\ntkrnlpa.exe, Win32 error 0n2 2. ...
- 新工具︱微软Microsoft Visual Studio的R语言模块下载试用Ing...(尝鲜)
笔者:前几天看到了以下的图片,着实一惊.作为R语言入门小菜鸟,还是觉得很好看,于是花了一点时间下载下来试用了一下,觉得还是挺高大上的. 就是英文不好是硬伤.下面贴给小白,我当时的下载步骤与遇见的问题. ...
- Caused by: java.lang.NoClassDefFoundError: org/hibernate/cfg/Configuration
1.错误描述 usage: java org.apache.catalina.startup.Catalina [ -config {pathname} ] [ -nonaming ] { -help ...
- Exynos4412交叉编译环境搭建
Exynos4412交叉编译环境搭建 交叉编译:在PC机(x86平台)上开发程序,在ARM板上运行,提高开发.编译速度. 环境: Tiny4412SDK1506开发板 需要软件: arm-linux- ...
- iOS - UIView 动画
1.UIView 动画 核心动画 和 UIView 动画 的区别: 核心动画一切都是假象,并不会真实的改变图层的属性值,如果以后做动画的时候,不需要与用户交互,通常用核心动画(转场). UIView ...
- ASP.NET VS2013 Office 转 PDF
本文适用于VS2013 项目中的Word转换为PDF.Excel转换为PDF.PPT转换为PDF 0.一种更加简单方便的方法 1.本页所用的方法在本机测试时基本不会出现问题,只是偶尔PPT转PDF失败 ...
- 前端(HTML/CSS/JS)-HTML编码规范
一.HTML编码规范 1. img标签要写alt属性 根据W3C标准,img标签要写alt属性,如果没有就写一个空的.但是一般要写一个有内容的,根据图片想要表达的意思,因为alt是在图片无法加载时显示 ...
- [ZJOI2007]时态同步
题目描述 小Q在电子工艺实习课上学习焊接电路板.一块电路板由若干个元件组成,我们不妨称之为节点,并将其用数字1,2,3….进行标号.电路板的各个节点由若干不相交的导线相连接,且对于电路板的任何两个节点 ...
- BZOJ2693jzptab
简单般Bzoj2154: Crash的数字表格 Sol 增加了数据组数T<=10000 推到 \(ans=\sum_{d=1}^{N}d*\sum_{i=1}^{\lfloor\frac{N}{ ...
- [BZOJ1087] [SCOI2005] 互不侵犯King (状压dp)
Description 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共8个格子. Input 只有一行,包 ...