zoukankan      html  css  js  c++  java
  • SPFA及SLF优化

    算法简介 
    SPFA(Shortest Path Faster Algorithm)是Bellman-Ford算法的一种队列实现,减少了不必要的冗余计算。 它可以在O(kE)的时间复杂度内求出源点到其他所有点的最短路径,可以处理负边。

    算法流程 
    SPFA对Bellman-Ford算法优化的关键之处在于意识到:只有那些在前一遍松弛中改变了距离估计值的点,才可能引起他们的邻接点的距离估计值的改变。因此,算法大致流程是用一个队列来进行维护,即用一个先进先出的队列来存放被成功松弛的顶点。初始时,源点s入队。当队列不为空时,取出队首顶点, 对它的邻接点进行松弛。如果某个邻接点松弛成功,且该邻接点不在队列中,则将其入队。经过有限次的松弛操作后,队列将为空,算法结束。SPFA算法的实现,需要用到一个先进先出的队列queue 和一个指示顶点是否在队列中的标记数组mark。为了方便查找某个顶点的邻接点,图采用邻接表存储

     

    判断负权回路的方案很多,世间流传最广、比较容易实现并且高效的方法的是记录每个结点进队次数,大于等于|V|次表示有负权。

     

    两个著名优化(SLF和LLL):

    SPFA 是按照 FIFO 的原则更新距离的, 没有考虑到距离标号的作用. 实现中 SPFA 有两个非常著名的优化: SLF 和 LLL. 

      SLF: Small Label First 策略. (比较常用)
      实现方法是, 设队首元素为 i, 队列中要加入节点 j, 在 d_j le d_i 时加到队首而不是队尾, 否则和普通的 SPFA 一样加到队尾. 

      LLL: Large Label Last 策略. (不太常用)
      实现方法是, 设队列 Q 中的队首元素为 i, 距离标号的平均值为 overline d = frac {sum_{j in Q} d_j }{left| Q right|}, 每次出队时, 若 d_i > overline d, 把 i 移到队列末尾, 如此反复, 直到找到一个 i 使 d_i le overline d, 将其出队. 

     

    C++模版:(没加SLF优化)

    #include <fstream>
    #include <iostream>
    #include <cstdio>
    #include <cstdlib>
    #include <cmath>
    #include <iomanip>
    #include <iomanip>
    #include <climits>
    #include <vector>
    #include <stack>
    #include <queue>
    #include <list>
    #include <set>
    #include <map>
    #include <algorithm>
    #include <string>
    #include <cstring>
    
    using namespace std;
    
    #define arraysize 501
    int maxData = 0x7fffffff;
    
    typedef struct edge
    {
       int to;
       int w;
    }edge;
    
    vector<edge> adjmap[arraysize]; //vector实现邻接表 
    int d[arraysize];
    bool final[arraysize];          //记录顶点是否在队列中,SPFA算法可以入队列多次 
    int cnt[arraysize];             //记录顶点入队列次数 
    
    bool SPFA(int s)
    {
         queue<int> myqueue;
         int i,j;
         for(i=0;i<n+1;++i)
         {
             d[i] = maxData;        //将除源点以外的其余点的距离设置为无穷大 
         }
         memset(final,0,sizeof(final));
         memset(cnt,0,sizeof(cnt));
         d[s]=0;                   //源点的距离为0 
         final[s] = true;          
         cnt[s]++;                 //源点的入队列次数增加 
         myqueue.push(s);
         int topint;
         while(!myqueue.empty())
         {
             topint = myqueue.front();myqueue.pop();
             final[topint] = false;
             for(i=0;i<adjmap[topint].size();++i)
             {
                 int to = adjmap[topint][i].to;
                 if(d[topint]<maxData && d[to]>d[topint]+ adjmap[topint][i].w)
                 {                                 
                      d[to] = d[topint]+ adjmap[topint][i].w;
                      if(!final[to])
                      {
                          final[to] = true;
                          cnt[to]++;
                          if(cnt[to]>=n)   //当一个点入队的次数>=n时就证明出现了负环。
                             return true;
                          myqueue.push(to);
                      }                  
                 }
             }
         }
         return false;
    }
    
    int main()
    {
        for(i=1;i<n+1;++i)       //此处特别注意对邻接表清空 
            adjmap[i].clear();
        for(i=0;i<m;++i)         //双向 
        {
            cin>>s>>e>>w;
            temp.to = e;
            temp.w = w;
            adjmap[s].push_back(temp);
            temp.to = s;
            adjmap[e].push_back(temp);      
        }
    }

     

    举杯独醉,饮罢飞雪,茫然又一年岁。 ------AbandonZHANG
  • 相关阅读:
    day01-h1字体大小和文本居中
    js正则表达式中的
    js滚动分页原理
    在web.xml中设置全局编码
    C# 导出word 表格代码
    C# 创建单例
    Winform 异步调用2 时间
    Winform 异步调用
    c#中跨线程调用windows窗体控件
    C# 中的委托和事件
  • 原文地址:https://www.cnblogs.com/AbandonZHANG/p/4113956.html
Copyright © 2011-2022 走看看