SPFA算法
  求单源最短路的SPFA算法的全称是:Shortest Path Faster
Algorithm
 最短路径快速算法-SPFA算法是西南交通大学段凡丁于1994年发表的。
  适用范围:给定的图存在负权边,这时类似Dijkstra等算法便没有了用武之地,而Bellman-Ford算法的复杂度又过高,SPFA算法便派上用场了。 我们约定有向加权图G不存在负权回路,即最短路径一定存在。当然,我们可以在执行该算法前做一次拓扑排序,以判断是否存在负权回路,但这不是我们讨论的重点。
  算法思想:我们用数组d记录每个结点的最短路径估计值,用邻接表来存储图G。我们采取的方法是动态逼近法:设立一个先进先出的队列用来保存待优化的结点,优化时每次取出队首结点u,并且用u点当前的最短路径估计值对离开u点所指向的结点v进行松弛操作,如果v点的最短路径估计值有所调整,v点不在当前的队列中,就将v点放入队尾。这样不断从队列中取出结点来进行松弛操作,直至队列空为止。


因为SPFA没有向迪杰斯塔拉算法那样,寻找dist[]的最小值,所以重复入队,更新dis[]的最小值,因为这个点本身dis[]的变化,会影响到与之邻接的点,所以要重复入队。
标程一:
const
int INF =
999999;
int 
map
[MAXN][MAXN];
//map[i,j]为初始输入的ij的距离,未知的map[i,j]=INF;
int 
dis
[MAXN];//源点Si的最短路
char vst[MAXN];//是否在队列中的标记
int     
Q[MAXN];
//队列
//
参数n表示结点数,s表示源点
int SPFA(int n,
int s)
{  int
i, pri, end, p, t;
// pri是队列头结点,end是队列尾结点
   
memset(vst,
0, sizeof(vst));//初始化
   
for(int
i=0; i<</b>MAXN; ++i)
       
Q
[i] =
0;//初始化队列为空
   
for (i=0; i<</b>n; i++)
       
dis
[i] =
INF;//初始化源点到I的值为最大值
   
dis
[s] =
0;//源点为0
   
vst
[s] =
1;//标记为已入队
Q[0] =
s;//源点入队
pri = 0; end
= 1;//队首队尾赋值
   
while (pri
<</b> end)
   
{
       
p
= Q[pri];//取队首元素
       
for (i=0; i<</b>n; ++i)
//更新dis
       
{ if (dis[p]+map[p][i] <</b> dis[i])
           
{ 
dis
[i] =
dis[p]+map[p][i];
               
if (!vst[i])    
//
未在队列中
               
{ 
Q
[end++]
= i;
                  
vst
[i] =
1;
               
}
           
}
       
}
       
vst
[p] =
0;   //

置出队的点为未标记

       
pri
++;
   
}
   
return 1;
}
标程二:
int num[999999]; //记录入队次数
 
void spfa(int s)  // 
初始结点s,即为起点,若目标结点t,则输出dict[t]。
{  
init_data(s);
   
int
head = 0, tail = 1;  
   
int path[Max];  // 
可增加一个path[]数组来记录最后s到t的路径。
    queue[0]
= s; //que.push(s);
 
   
dict[s] = 0;
   
while
(tail >
head)//(!que.empty())
 
    {
int
u = queue[head];//int
u=que.front();  
//que.pop();
       
vis[u] = true;
 
       
for
(i = 1; i <= n; i ++)
       
{ if
(dict[i] > dict[u] +
edge[u][i])
           
{  dict[i] = dict[u] +
edge[u][i];
              
path[i] = u;
 
              
num[i]++
              
if(num[i]>=n) return
1;//判断是否有负权值……
              
if
(!vis[i]) 
//  对以前没有访问过的顶点,加入队列中。
               
{  vis[i] = true;
                  
queue[tail] = i;// que.push(i);
                  
tail
++;                      

 
               
}
            
}
        
}
        
vis[u] = false;  // 
记得把出队列的顶点的vis[]置为false。
       
head ++;
 
   
}
}
判断负权回路
num[i]>=n的原因,即使所有的点更新都会让i入队的话,才只有n-1次,这时一定是最小值了,入队n次,一定有负权回路

最短路径:我的理解--SPFA算法的更多相关文章

  1. SPFA算法——最短路径

    粗略讲讲SPFA算法的原理,SPFA算法是1994年西南交通大学段凡丁提出 是一种求单源最短路的算法 算法中需要用到的主要变量 int n;  //表示n个点,从1到n标号 int s,t;  //s ...

  2. 【算法】单元最短路径之Bellman-Ford算法和SPFA算法

    SPFA是经过对列优化的bellman-Ford算法,因此,在学习SPFA算法之前,先学习下bellman-Ford算法. bellman-Ford算法是一种通过松弛操作计算最短路的算法. 适用条件 ...

  3. (转)SPFA算法

    原文地址:http://www.cnblogs.com/scau20110726/archive/2012/11/18/2776124.html 粗略讲讲SPFA算法的原理,SPFA算法是1994年西 ...

  4. SPFA 算法(剪辑)(学习!)

    SPFA算法 单源最短路径的算法最常用的是Dijkstra,些算法从时间复杂度来说为O(n^2),但是面对含有负权植的图来说就无能为力了,此时 Dellman-ford算法就有用了,这咱算法是采用的是 ...

  5. 转载:SPFA算法学习

    转载地址:http://www.cnblogs.com/scau20110726/archive/2012/11/18/2776124.html 粗略讲讲SPFA算法的原理,SPFA算法是1994年西 ...

  6. 数据结构与算法--最短路径之Bellman算法、SPFA算法

    数据结构与算法--最短路径之Bellman算法.SPFA算法 除了Floyd算法,另外一个使用广泛且可以处理负权边的是Bellman-Ford算法. Bellman-Ford算法 假设某个图有V个顶点 ...

  7. (最短路径算法整理)dijkstra、floyd、bellman-ford、spfa算法模板的整理与介绍

    这一篇博客以一些OJ上的题目为载体.整理一下最短路径算法.会陆续的更新... 一.多源最短路算法--floyd算法 floyd算法主要用于求随意两点间的最短路径.也成最短最短路径问题. 核心代码: / ...

  8. 最短路径问题的Dijkstra和SPFA算法总结

    Dijkstra算法: 解决带非负权重图的单元最短路径问题.时间复杂度为O(V*V+E) 算法精髓:维持一组节点集合S,从源节点到该集合中的点的最短路径已被找到,算法重复从剩余的节点集V-S中选择最短 ...

  9. 最短路径--SPFA 算法

    适用范围:给定的图存在负权边,这时类似Dijkstra等算法便没有了用武之地,而Bellman-Ford算法的复杂度又过高,SPFA算法便派上用场了. 我们约定有向加权图G不存在负权回路,即最短路径一 ...

随机推荐

  1. 数组中的each 和 jquery 中的 each

    数组的实例上都有一个叫做 forEach 的方法,这个方法定义在 Array.prototype 上,所以数组的所有实例都可以使用 forEach 这个方法. forEach 方法的语法结构如下: v ...

  2. unity3d 资源文件从MAX或者MAYA中导出的注意事项

    unity3d 资源文件从MAX或者MAYA中导出的注意事项     1.首先,Unity3d 中,导出带动画的资源有2种导出方式可以选择:    1) 导出资源时,只导出一个文件,保留模型,骨骼和所 ...

  3. flask插件系列之SQLAlchemy基础使用

    sqlalchemy是一个操作关系型数据库的ORM工具.下面研究一下单独使用和其在flask框架中的使用方法. 直接使用sqlalchemy操作数据库 安装sqlalchemy pip install ...

  4. shell脚本自带变量的含义

    $0 Shell本身的文件名 $1-$n 添加到Shell的各参数值.$1是第1参数.$2是第2参数… $$ Shell本身的PID(ProcessID) $! Shell最后运行的后台Process ...

  5. python从2.6.x升级到2.7.x

    [前提] 今日是20171207,目前Linux发行版默认安装的Python版本都是2.6.x,但是这个版本Python已经不再进行维护了. 所以需要将Python做一个升级,到2.7.x [注意] ...

  6. Nginx源码分析-ngx_module_s结构体

    该结构体是整个Nginx模块化架构最基本的数据结构体.它描述了Nginx程序中一个模块应该包括的基本属性,在tengine/src/core/ngx_conf_file.h中定义了该结构体 struc ...

  7. java版云笔记(六)之AOP

    今天主要是利用aop技术追加service的响应时间的计算和异常的日志记录. AOP AOP(Aspect Oriented Programming),即面向切面编程,可以说是OOP(Object O ...

  8. css绘制图标

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xht ...

  9. hdu 5918(强行水过去..正解KMP)

    Sequence I Time Limit: 3000/1500 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total ...

  10. fastdfs5.11+centos7.2 按照部署(一)【转载】

    1.绪论 最近要用到fastDFS,所以自己研究了一下,在搭建FastDFS的过程中遇到过很多的问题,为了能帮忙到以后搭建FastDFS的同学,少走弯路,与大家分享一下.FastDFS的作者淘宝资深架 ...