SPFA研究

说起来,SPFA这个算法之前一直是半懂不懂,偶尔敲一个模板,然后囫囵做个两三道水题,倒也是轻松,然而一旦遇到比较复杂的题目的时候总是摸不了门道,于是下了决心花几个小时一定要把它吃透。不过没有想到深入探究了一番,其实SPFA也不是很复杂,结合之前的最短路问题求解的相关知识,很快就能掌握其中的技巧,甚是欣喜。

简介:

SPFA就是使用队列或者栈的Bellman-Ford算法的优化版本。都知道Bellman-Ford算法的复杂度之高,不是在所有时候都适用。SPFA正是为了解决这个问题而生的。

SPFA(Shortest Path Faster Algorithm)算法是求单源最短路径的一种算法,它是Bellman-ford的队列优化,它是一种十分高效的最短路算法。

很多时候,给定的图存在负权边,这时类似Dijkstra等算法便没有了用武之地,而Bellman-Ford算法的复杂度又过高,SPFA算法便派上用场了。SPFA的复杂度大约是O(kE),k是每个点的平均进队次数(一般的,k是一个常数,在稀疏图中小于2)。

但是,SPFA算法稳定性较差,在稠密图中SPFA算法时间复杂度会退化。

实现方法:建立一个队列,初始时队列里只有起始点,在建立一个表格记录起始点到所有点的最短路径(该表格的初始值要赋为极大值,该点到他本身的路径赋为0)。然后执行松弛操作,用队列里有的点去刷新起始点到所有点的最短路,如果刷新成功且被刷新点不在队列中则把该点加入到队列最后。重复执行直到队列为空。

此外,SPFA算法还可以判断图中是否有负权环,即一个点入队次数超过N。

参考
SPFA算法详解
最短路径 之 SPFA算法

常用数组

struct Node
{
    int to;                     //指向点标号
    int cost;                   //路径长度
    int next;                   //下一个点标号
}node[max_edges];               //max_edges为最大边数,储存每一边状态

int head[max_nodes];            //max_nodes为最大点数,储存邻接表头指针位置
int vis[max_nodes];             //记录节点在队列中是否存在
int dis[max_nodes];             //储存最短路径长度

队列式SPFA(常见)

void SPFA(int start,int n)      //SPFA算法核心,start为起点标号,n为节点数量
{
    memset(vis,0,sizeof(vis));             //清空访问标记
    for ( int i = 1 ; i <= n ; i++){       //初始化距离为INF
          d[i] = INF;
    }
    dis[start] = 0                //将起始位置距离设置为0
    queue<int>Q;
    Q.push(start);                //将初始位置推入队列中
    vis[start] = 1;               //标记在队列中初始位置出现过

    while(!Q.empty())
    {
        int x = Q.front();
        Q.pop();
        vis[x] = 0;             //将节点从队列退出后要记得将访问标记设置为0

        for(int i = f[x] ; i != -1 ; i = next[i])   //访问该节点下的所有分支节点
        {
            int t = node[i].to;
            if(d[t] > d[x] + node[i].cost)    //松弛操作
            {
                d[t] = d[x] + node[i].cost;
                if(!vis[t])                //判断节点是否在队列中
                {
                    Q.push(t);
                    vis[t] = 1;
                }
            }
        }
    }
}

添加操作

void add(int u,int v,int c)
{
    node[cnt].to = v ;//指向节点
    node[cnt].cost = c;//路径长度
    node[cnt].next = head[u] ;  //构建邻接表
    head[u] = cnt++;
}

初始化操作

void init()
{
     int cnt = 0;                 //cnt为边的数量,添加操作
    memset(head,-1,sizeof(head));         
 //初始化头指针位置为-1方便SPFA中对于每一个节点搜索下一个节点的结束
}

栈型SPFA(占用内存更少,运行效率更高,用数组模拟更佳)

void SPFA(int sta,int n)
{
    stack<int>S;
    for(int i = 1 ; i <= n ; i++){
        if(i == sta){
            S.push(i);
            vis[i] = true;
            dis[i] = 0;
        }
        else
        {
            vis[i] = false;
            dis[i] = INF;
        }
    }

    while(!S.empty())
    {
        int u = S.top();
        S.pop();
        vis[u] = false;
        for(int i=head[u];i!=-1;i=edge[i].next)
        {
            int v=edge[i].to;
            if(dis[v] > dis[u] + edge[i].v)
            {
                dis[v] = dis[u] + edge[i].v;
                if(!vis[v]){
                    vis[v] = true;
                    S.push(v);
                }
            }
        }
    }
}
最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容

  • 回想起曾学习A-star寻径算法时,难以透彻理解其原理和机制,但随着对图和搜索算法的理解愈发深入,近期重拾A-st...
    胡哈哈哈阅读 4,232评论 0 1
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • 五月第一天是劳动节,放假。总觉得是个好兆头,因为是个可以开始某样习惯的日子(虽然任何一天都可以:),可以适当总结总...
    宋逸斌阅读 466评论 0 1
  • 01 曾经我一直以为,这样的事情我只会在新闻中看到,就像隔着无数座大山那样遥远…… 朋友陪她一起去打的胎,这个仅仅...
    巅疯娜阅读 620评论 6 14
  • 当前天数:第78天。 剩余天数:22天。 参与人:我。 路程:5圈。 地点:家里小区的跑步场。 开始时间:22点2...
    Teratimes成长论阅读 202评论 0 0