为了不要让太多人被害,我还是说一下这种算法,它实际上很简单,但被人讲着讲着绕晕了。

主要思想

有人说,SPFA是Bellman-Ford的队列优化。这个算法我也懂了,但是还没试过。我不管是什么算法的优化,反正我看着不像。

它的思想很简单:BFS。有人说这只是类似的,并不是纯BFS。我不管这些,分这么严格干嘛呢!

从起点开始,枚举它节点的边,走所有与它相连的路径。如果能更新别的节点就更新,不能更新嘛,就直接将它从队列里删掉,不要它了,反正它没鬼用。

还有一点,要标记某节点是否已经在队列里,将一个节点加入时,看看它在不在队列里面,如果在,就不用放进去,反正没鬼用。

若某个点在队列里出现过N次,则是进入了负环,赋个无限小就行了。

实现讲解

先讲讲图的存储。这个图并不是用邻接矩阵来弄的(你也能用,速度会慢,也许还会出现溢出等奇怪的错误)。这个东西叫邻接表,有的人也叫它边集数组。就是记录每一个节点,与它相连的所有的边。每条边的信息有这个点的另一端,已及这条边的权值。

我是这样定义的

struct _Way//定义_Way类型,表示某点与y点相连,长度为len
{
int y,len;
};
_Way way[1001][1001] {};//way[i][j]标示i的第j条路
int now[1001] {};//now[i]即为节点i所连的边的数量 Number of ways

当然,你也可以开动态数组。不过如果不是急需省内存,你就开吧。但是我没试过,也许会更繁琐。

读入时也许读入重边,用指针判断判断就好了

正常的BFS不用说了吧。。。

判断一个节点是否在队列中,只需要一个数组标记就好了。

具体代码

#include <iostream>
#include <string.h>
#include <math.h>
#include <limits.h>
using namespace std;
struct _Way//定义_Way类型,表示某点与y点相连,长度为len
{
int y,len;
};
int n,m;
int q;
_Way way[1001][1001] {};//way[i][j]标示i的第j条路
_Way* bz[1001][1001] {};//输入时用于判断,有时输入了重边,肯定要取最小的,这个标记了这个边的地址,方便判断
int now[1001] {};//now[i]即为节点i所连的边的数量 Number of ways
int dis[1001] {};//dis[i]即为从起点到i的最短距离
int min(int a,int b){return a<b?a:b;}//取最小值函数
void SPFA(int);
int main()
{
ios::sync_with_stdio(false);
cin>>n>>m;
for(int i=1;i<=m;i++)
{
int x,y,len;
cin>>x>>y>>len;
if (bz[x][y]!=NULL)//判断这条边是否重了
{
bz[x][y]->len=min(bz[x][y]->len,len);//如果重了边取最小值
continue;
}
now[x]++;
way[x][now[x]].y=y;
way[x][now[x]].len=len;
bz[x][y]=&(way[x][now[x]]);//将它的地址赋给bz[x][y]
}
cin>>q;
SPFA(q);
for(int i=1;i<=n;i++)
{
if (dis[i]==INT_MAX) cout<<"-1"<<endl;//如果到不了输出-1
else cout<<dis[i]<<endl;
}
return 0;
}
void SPFA(int q)//q表示起点
#define SIZE 2006
{
bool bz[1001] {};//bz[i]表示i节点是否在队列中
for(int i=1;i<=n;i++)
dis[i]=INT_MAX;//初始化为无限大
int head=0,tail=1;
int d[SIZE+1];
dis[q]=0;//起点到起点,当然为0了
d[1]=q;
bz[q]=1;
do
{
head++;
if (head>SIZE) head=1;//与下文一样,用循环队列
for(int i=1;i<=now[d[head]];i++)//将跟它相连的边都枚举一遍
{
tail++;
if (tail>SIZE) tail=1;
d[tail]=way[d[head]][i].y;
if (dis[d[tail]]<=dis[d[head]]+way[d[head]][i].len)//如果现在到那边不比之前的优,则删掉它
{
tail--;
if (tail<1) tail=SIZE;
continue;
}
dis[d[tail]]=dis[d[head]]+way[d[head]][i].len;//更新
if (bz[d[tail]])
{
tail--;
if (tail<1) tail=SIZE;
continue;
}
bz[d[tail]]=1;//标记其以进入队列
}
bz[d[head]]=0;//它出了队列,将之前的1改为0
}
while (head!=tail);
}

优化

C++者勿入

图论最短路径算法——SPFA的更多相关文章

  1. 图论最短路径算法总结(Bellman-Ford + SPFA + DAGSP + Dijkstra + Floyd-Warshall)

    这里感谢百度文库,百度百科,维基百科,还有算法导论的作者以及他的小伙伴们...... 最短路是现实生活中很常见的一个问题,之前练习了很多BFS的题目,BFS可以暴力解决很多最短路的问题,但是他有一定的 ...

  2. 图论最短路径算法——Dijkstra

    说实在的,这算法很简单,很简单,很简单--因为它是贪心的,而且码量也小,常数比起SPFA也小. 主要思想 先初始化,dis[起点]=0,其它皆为无限大. 还要有一个bz数组,bz[i]表示i是否确定为 ...

  3. 几大最短路径算法比较(Floyd & Dijkstra & Bellman-Ford & SPFA)

    几个最短路径算法的比较:Floyd 求多源.无负权边(此处错误?应该可以有负权边)的最短路.用矩阵记录图.时效性较差,时间复杂度O(V^3).       Floyd-Warshall算法(Floyd ...

  4. 几个最短路径算法Floyd、Dijkstra、Bellman-Ford、SPFA的比较

        几大最短路径算法比较 转自:http://blog.csdn.net/v_july_v/article/details/6181485 几个最短路径算法的比较: Floyd        求多 ...

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

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

  6. 几个最短路径算法Floyd、Dijkstra、Bellman-Ford、SPFA的比较(转)

    几大最短路径算法比较 几个最短路径算法的比较:Floyd        求多源.无负权边(此处错误?应该可以有负权边)的最短路.用矩阵记录图.时效性较差,时间复杂度O(V^3).       Floy ...

  7. 图论-最短路径 2.Dijkstra算法O (N2)

    2.Dijkstra算法O (N2) 用来计算从一个点到其他所有点的最短路径的算法,是一种单源最短路径算法.也就是说,只能计算起点只有一个的情况. Dijkstra的时间复杂度是O (N2),它不能处 ...

  8. Cocos2d-x 地图步行实现1:图论Dijkstra算法

    下一节<Cocos2d-x 地图行走的实现2:SPFA算法>: http://blog.csdn.net/stevenkylelee/article/details/38440663 本文 ...

  9. [联赛可能考到]图论相关算法——COGS——联赛试题预测

    COGS图论相关算法 最小生成树 Kruskal+ufs int ufs(int x) { return f[x] == x ? x : f[x] = ufs(f[x]); } int Kruskal ...

随机推荐

  1. centos7.2搭建kubernetes1.5.2+dashboard

    一.    简介 近来容器对企业来说已经不是什么陌生的概念,Kubernetes作为Google开源的容器运行平台,受到了大家的热捧.搭建一套完整的kubernetes平台,也成为试用这套平台必须迈过 ...

  2. RouterOS视频教程下载

    下载信息 名称:RouterOS视频教程下载 格式:MP4 版本:V1.0 https://pan.baidu.com/s/1skU6oW1 下载密码:nb97

  3. super 关键字的使用及说明

    super 关键字主要用于访问父类的变量和方法. 代码示例: public class Student { String name; public Student(){ System.out.prin ...

  4. RHEL7更换yum源

    1,删除注册和订阅提示 查找subscription-manager相关组件: rpm -qa | grep subscription-manager 删除subscription-maanager相 ...

  5. HTTPS客户端的java实现

    目录 https客户端 指定ssl算法套 浏览器可以作为客户端向服务器发送Http请求,当需要访问后台或第三方Restful接口时,也可以用java实现客户端直接发get/post请求,获取数据. h ...

  6. 312. 戳气球【困难】【区间DP】

    题目链接 有 n 个气球,编号为0 到 n-1,每个气球上都标有一个数字,这些数字存在数组 nums 中. 现在要求你戳破所有的气球.每当你戳破一个气球 i 时,你可以获得 nums[left] * ...

  7. 关于python merge后数据行数增加的问题

    其中一个可能的原因是 join 的 data 里面的列不唯一,也就是要匹配的表里面有些一行数据对应了被匹配表多条数据,这样出来可能会增加行数,可以再查一下被匹配表里的数据是否去重

  8. c结构体指针使用

    #include <stdio.h> #include<stdlib.h> #include<string.h> typedef struct _Date { un ...

  9. java 堆排,优先级队列,归并排序

    堆排 堆排是基于二叉树而得来的 例如:对一个数组 可以转为二叉树:       二叉树特性父节点为 i ,  左叶子节点为2i+1:右叶子节点为2i+2; 步骤分解: 1. 先从第一个非叶子节点(即下 ...

  10. 挂载U盘

    .fdisk -l 查看当前系统存储盘 (sdaX一般是系统自带, sdbX则是外接) .mount /dev/sdbX /mnt/usb/ (如果usb目录不存在可创建新目录) .umount /m ...