最短路算法详解(Dijkstra/SPFA/Floyd)
新的整理版本版的地址见我新博客 http://www.hrwhisper.me/?p=1952
一、Dijkstra
使用邻接矩阵的时间复杂度为O(n^2),用优先队列的复杂度为O((m+n)logn)近似为O(mlogn)
(一) 过程
每次选择一个未访问过的到已经访问过(标记为Known)的所有点的集合的最短边,并用这个点进行更新,过程如下:
Dv为最短路,而Pv为前面的顶点。
1. 初始
|
V |
Known |
Dv |
Pv |
|
V1 |
F |
0 |
0 |
|
V2 |
F |
∞ |
0 |
|
V3 |
F |
∞ |
0 |
|
V4 |
F |
∞ |
0 |
|
V5 |
F |
∞ |
0 |
|
V6 |
F |
∞ |
0 |
|
V7 |
F |
∞ |
0 |
2. 在v1被标记为已知后的表
|
V |
Known |
Dv |
Pv |
|
V1 |
T |
0 |
0 |
|
V2 |
F |
2 |
V1 |
|
V3 |
F |
∞ |
0 |
|
V4 |
F |
1 |
V1 |
|
V5 |
F |
∞ |
0 |
|
V6 |
F |
∞ |
0 |
|
V7 |
F |
∞ |
0 |
3. 下一步选取v4并且标记为known,顶点v3,v5,v6,v7是邻接的顶点,而他们实际上都需要调整。如表所示:
|
V |
Known |
Dv |
Pv |
|
V1 |
T |
0 |
0 |
|
V2 |
F |
2 |
V1 |
|
V3 |
F |
3 |
V4 |
|
V4 |
T |
1 |
V1 |
|
V5 |
F |
3 |
V4 |
|
V6 |
F |
9 |
V4 |
|
V7 |
F |
5 |
V4 |
4. 接下来选取v2,v4是邻接点,但已经是known的,不需要调整,v5是邻接的点但不做调整,因为经过v2的值为2+10=12而长为3的路径已经是已知的。
|
V |
Known |
Dv |
Pv |
|
V1 |
T |
0 |
0 |
|
V2 |
T |
2 |
V1 |
|
V3 |
F |
3 |
V4 |
|
V4 |
T |
1 |
V1 |
|
V5 |
F |
3 |
V4 |
|
V6 |
F |
9 |
V4 |
|
V7 |
F |
5 |
V4 |
5. 接下来选取v5,值为3,v7 3+6>5不需调整,然后选取v3,对v6的距离下调到3+5=8
|
V |
Known |
Dv |
Pv |
|
V1 |
T |
0 |
0 |
|
V2 |
T |
2 |
V1 |
|
V3 |
T |
3 |
V4 |
|
V4 |
T |
1 |
V1 |
|
V5 |
T |
3 |
V4 |
|
V6 |
F |
8 |
V3 |
|
V7 |
F |
5 |
V4 |
6. 再选下一个顶点是v7,v6变为5+1=6
|
V |
Known |
Dv |
Pv |
|
V1 |
T |
0 |
0 |
|
V2 |
T |
2 |
V1 |
|
V3 |
T |
3 |
V4 |
|
V4 |
T |
1 |
V1 |
|
V5 |
T |
3 |
V4 |
|
V6 |
F |
6 |
V7 |
|
V7 |
T |
5 |
V4 |
7. 最后选取v6
|
V |
Known |
Dv |
Pv |
|
V1 |
T |
0 |
0 |
|
V2 |
T |
2 |
V1 |
|
V3 |
T |
3 |
V4 |
|
V4 |
T |
1 |
V1 |
|
V5 |
T |
3 |
V4 |
|
V6 |
T |
6 |
V7 |
|
V7 |
T |
5 |
V4 |
(二) 局限性
Dijkstra没办法解决负边权的最短路径,如图
运行完该算法后,从顶点1到顶点3的最短路径为1,3,其长度为1,而实际上最短路径为1,2,3,其长度为0.(因为过程中先选择v3,v3被标记为已知,今后不再更新)
(三) 算法实现。
1.普通的邻接表 以(HDU 1874 畅通工程续 SPFA || dijkstra)为例
用vis作为上面标记的known,dis记录最短距离(记得初始化为一个很大的数)。
void dijkstra(int s)
{
memset(vis,0,sizeof(vis));
int cur=s;
dis[cur]=0;
vis[cur]=1;
for(int i=0;i<n;i++)
{
for(int j=0;j<n;j++)
if(!vis[j] && dis[cur] + map[cur][j] < dis[j]) //未被标记且比已知的短,可更新
dis[j]=dis[cur] + map[cur][j] ; int mini=INF;
for(int j=0;j<n;j++)
if(!vis[j] && dis[j] < mini) //选择下一次到已知顶点最短的点。
mini=dis[cur=j];
vis[cur]=true;
}
}
2.邻接表+优先队列。
要重载个比较函数.
struct point
{
int val,id;
point(int id,int val):id(id),val(val){}
bool operator <(const point &x)const{
return val>x.val;
}
};
void dijkstra(int s)
{
memset(vis,0,sizeof(vis));
for(int i=0;i<n;i++)
dis[i]=INF; priority_queue<point> q;
q.push(point(s,0));
dis[s]=0;
while(!q.empty())
{
int cur=q.top().id;
q.pop();
if(vis[cur]) continue;
vis[cur]=true;
for(int i=head[cur];i!=-1;i=e[i].next)
{
int id=e[i].to;
if(!vis[id] && dis[cur]+e[i].val < dis[id])
{
dis[id]=dis[cur]+e[i].val;
q.push(point(id,dis[id]));
}
}
}
}
二、SPFA(bellman-ford)
(一)原理过程:
(二)实现:
void SPFA(int s)
{
for(int i=0;i<n;i++)
dis[i]=INF; bool vis[MAXN]={0}; vis[s]=true;
dis[s]=0; queue<int> q;
q.push(s);
while(!q.empty())
{
int cur=q.front();
q.pop();
vis[cur]=false;
for(int i=0;i<n;i++)
{
if(dis[cur] + map[cur][i] < dis[i])
{
dis[i]=dis[cur] + map[cur][i];
if(!vis[i])
{
q.push(i);
vis[i]=true;
}
}
}
}
}
void spfa(int s)
{
memset(vis,0,sizeof(vis));
for(int i=0;i<n;i++)
dis[i]=INF; queue<int> q;
q.push(s);
vis[s]=true;
dis[s]=0;
while(!q.empty())
{
int cur=q.front();
q.pop();
vis[cur]=false;
for(int i=head[cur];i!=-1;i=e[i].next)
{
int id=e[i].to;
if(dis[id] > dis[cur]+e[i].val)
{
dis[id] = dis[cur] + e[i].val;
if(!vis[id])
{
vis[id]=true;
q.push(id);
}
}
}
}
}
3.上面的两个都没有对负圈的判断,因为题目的限制就是正的。判断负环代码如下:以(ZOJ 2770 Burn the Linked Camp 差分约束)为例
bool spfa()
{
for(int i=0;i<=n;i++)
dis[i]=INF; bool vis[MAXN]={0};
int cnt[MAXN]={0};
queue<int> q;
dis[0]=0;
vis[0]=true;
cnt[0]=1;
q.push(0); while(!q.empty())
{
int cur=q.front();
q.pop();
vis[cur]=false; for(int i=head[cur];i!=-1;i=e[i].next)
{
int id=e[i].to;
if(dis[cur] + e[i].val > dis[id])
{
dis[id]=dis[cur]+e[i].val;
if(!vis[id])
{
cnt[id]++;
if(cnt[cur] > n)
return false;
vis[id]=true;
q.push(id);
}
}
}
}
return true;
}
(三):优化
(四)应用:
三、floyd
(一)原理过程:
(二)实现:
void floyd()
{
for(int k=0;k<n;k++)
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
dis[i][j]=min(dis[i][j],dis[i][k]+dis[k][j]);
}
四、其他
如走迷宫经常用的BFS,以一个点出发,向外扩散。
如:
除了上面的
HDU 1874畅通工程续 SPFA || dijkstra||floyd
还有:
UVA11280 - Flying to Fredericton SPFA变形
UVA11090 - Going in Cycle!! SPFA
UVA10917 Walk Through the Forest SPFA
POJ 3259Wormholes邻接表的SPFA判断负权回路
POJ 1932XYZZY (ZOJ 1935)SPFA+floyd
UVA11374 Airport Express SPFA||dijkstra
UVA11367 - Full Tank? dijkstra+DP
POJ 1511Invitation Cards (ZOJ 2008)使用优先队列的dijkstra
POJ 3268Silver Cow Party (Dijkstra~)
POJ 2387Til the Cows Come Home (Dijkstra)
最短路算法详解(Dijkstra/SPFA/Floyd)的更多相关文章
- 最短路算法详解(Dijkstra,Floyd)
最短路径 在一个无权的图中,若从一个顶点到另一个顶点存在着一条路径,则称该路径长度为该路径上所经过的边的数目,它等于该路径上的顶点数减1.由于从一个顶点到另一个顶点可能存在着多条路径,每条路径上所经过 ...
- 【最短路径Floyd算法详解推导过程】看完这篇,你还能不懂Floyd算法?还不会?
简介 Floyd-Warshall算法(Floyd-Warshall algorithm),是一种利用动态规划的思想寻找给定的加权图中多源点之间最短路径的算法,与Dijkstra算法类似.该算法名称以 ...
- BM算法 Boyer-Moore高质量实现代码详解与算法详解
Boyer-Moore高质量实现代码详解与算法详解 鉴于我见到对算法本身分析非常透彻的文章以及实现的非常精巧的文章,所以就转载了,本文的贡献在于将两者结合起来,方便大家了解代码实现! 算法详解转自:h ...
- kmp算法详解
转自:http://blog.csdn.net/ddupd/article/details/19899263 KMP算法详解 KMP算法简介: KMP算法是一种高效的字符串匹配算法,关于字符串匹配最简 ...
- 机器学习经典算法详解及Python实现--基于SMO的SVM分类器
原文:http://blog.csdn.net/suipingsp/article/details/41645779 支持向量机基本上是最好的有监督学习算法,因其英文名为support vector ...
- [转] KMP算法详解
转载自:http://www.matrix67.com/blog/archives/115 KMP算法详解 如果机房马上要关门了,或者你急着要和MM约会,请直接跳到第六个自然段. 我们这里说的K ...
- 【转】AC算法详解
原文转自:http://blog.csdn.net/joylnwang/article/details/6793192 AC算法是Alfred V.Aho(<编译原理>(龙书)的作者),和 ...
- KMP算法详解(转自中学生OI写的。。ORZ!)
KMP算法详解 如果机房马上要关门了,或者你急着要和MM约会,请直接跳到第六个自然段. 我们这里说的KMP不是拿来放电影的(虽然我很喜欢这个软件),而是一种算法.KMP算法是拿来处理字符串匹配的.换句 ...
- EM算法详解
EM算法详解 1 极大似然估计 假设有如图1的X所示的抽取的n个学生某门课程的成绩,又知学生的成绩符合高斯分布f(x|μ,σ2),求学生的成绩最符合哪种高斯分布,即μ和σ2最优值是什么? 图1 学生成 ...
随机推荐
- java导入大量Excel时报错
在项目中同事遇到一问题,如今给大家分享一下. 在程序里面导入两千多条数据后.程序就报错. 刚開始以为是内存的问题.在经过细致跟踪代码后发现每次都是833行的第三列报错.也就是第一万列.最后在网上找到了 ...
- 自己定义控件-DownSlidingDrawer
一.描写叙述 能够下拉的 SlidingDrawer 二.效果图 图片是网上找到,可是效果是一样的 三.源代码 https://github.com/mentor811/Demo_MySlidingD ...
- ajax ---- json 和 xml 区别
2.XML和JSON优缺点 (1).XML的优缺点<1>.XML的优点 A.格式统一,符合标准: B.容易与其他系统进行远程交互,数据共享比较方便.<2>.XML的缺点 A.X ...
- 73,QT指针数组实战(指针数组与数组指针)
//指针数组,每一个指针都是一个MainWindow // MainWindow *w[3][4]; // for(int i=0;i<3;i++) // { // for(int j=0;j& ...
- DbSet<>.Find()
第一篇为大家带来新的API,DbSet<>.Find(). 过去我们常常用Where或First(FirstOrDefault)方法来查找对应的实体,比如: var people = fr ...
- Kinect开发 —— 基础知识
转自:http://www.cnblogs.com/yangecnu/archive/2012/04/02/KinectSDK_Application_Fundamentals_Part2.html ...
- 关于css的入门知识
css:叠层样式表,给html添加样式的 接下来说一说,在网页中如何嵌套style样式 1.行间样式:把style(*权重1000)作为属性卸载标签里 eg:<p style="col ...
- vsphere client和vsphere web client的区别
vsphere client是一个运行在windows桌面上的客户端,在linux环境下无法运行,在vsphere5.0以后,VMware在逐渐弱化vsphere client的作用,现在很多高级功能 ...
- Java Reflection - Getters and Setters
原文链接:http://tutorials.jenkov.com/java-reflection/getters-setters.html 通过使用 Java 反射,我们能够在程序执行时观察 clas ...
- Codeforces 441 B. Valera and Fruits
B. Valera and Fruits time limit per test 1 second memory limit per test 256 megabytes input standard ...