最短路径 SPFA P3371 【模板】单源最短路径(弱化版)

P3371 【模板】单源最短路径(弱化版)

SPFA算法:

       SPFA 算法是 Bellman-Ford算法 的队列优化算法的别称,通常用于求含负权边的单源最短路径,以及判负权环。SPFA 最坏情况下复杂度和朴素 Bellman-Ford 相同,为 O(VE)。

 

 

SPFA和Dijkstra不同的是:

  Dijkstra  是从一个点的所有出边中找到一个最短出边,用它来继续更新下边的点

   SPFA     是用一个点的所有出边都更新它下面的点

                 更新之前把这个点存进队列

                 更新时把他拿出来,再把更新的出边终点(未入队的)入队

                 一直不断更新,直到队列为空

                 队列里存的是

 (下面有详细解释,在链式前向星以后)

 

 

head[---]     这里大小根据点数决定
                    记录存边的历史,存的是i点的最后一条出边(它经历了不断更新)

 vis[---]         判断是否已存入队列

 dis[---]         从起点开始到当前点的最短路径

 num_edge  表示边的编号

 

 

 

这里要用链式前向星存图:

 

复制代码
//以下为链式前向星存图 
void addedge(int from,int to,int dis)   //存储每一条边 :起点,终点,长度
{
    num_edge++;                         //新建一条边 
    edge[num_edge].next=head[from];     //上一条出边 
    edge[num_edge].to=to;
    edge[num_edge].dis=dis;
    head[from]=num_edge;                //记录最后一条出边 
}
复制代码

 这里edge[1]=0,因为它是顶点1的第一条出边

        edge[2]=1,edge[3]=2,

        edge[5]=0,因为它是顶点5 的第一条出边

        edge[7]=5

 

 

 

SPFA

默认起点是1

 用到1就把它弹出再用6更新5入队再用3更新

 

直到队列为空

 

 

 

 

 

【代码】:

复制代码
#include<bits/stdc++.h>
#include<queue>
using namespace std;

const int inf=2147483647;

int n,m,s;
int dis[10008],vis[10008],head[10008],num_edge;

struct Edge
{
    int next,to,dis;
}edge[500008];  //大小由边数决定
// to   目标点
// dis  权值
// next 该点的上一条出边

queue<int>q;

//以下为链式前向星存图 
void addedge(int from,int to,int dis) //存储每一条边 : 起点,终点,长度 { num_edge++; //新建一条边 edge[num_edge].next=head[from]; //上一条出边 edge[num_edge].to=to; edge[num_edge].dis=dis; head[from]=num_edge; //记录最后一条出边 } void spfa() { for(int i=1;i<=n;i++) { dis[i]=inf; //初始化最大值 vis[i]=0; //都不入队 } dis[s]=0; vis[s]=1; q.push(s); //把起点S入队 while(!q.empty()) { int u=q.front(); //当前起点 q.pop(); //用就弹出 vis[u]=0; //弹出后记录为不在队列 for(int i=head[u];i;i=edge[i].next) //遍历起点的所有出边 { int v=edge[i].to; //当前终点 if(dis[v]>dis[u]+edge[i].dis)
//如果【起点到当前终点的距离】>【起点到当前起点的距离+当前距离与当前终点距离】
//那就更新为更小距离
{ dis[v]
=dis[u]+edge[i].dis; if(!vis[v]) //未入队的当前终点入队 { q.push(v); vis[v]=1; } } } } } int main() { scanf("%d%d%d",&n,&m,&s); for(int i=1;i<=m;i++) { int u,v,w; scanf("%d%d%d",&u,&v,&w); addedge(u,v,w); } spfa(); for(int i=1;i<=n;i++) { if(i==s) printf("0 "); else printf("%d ",dis[i]); } return 0; }
复制代码

 

posted @   晔子  阅读(448)  评论(0编辑  收藏  举报
编辑推荐:
· 深入理解 Mybatis 分库分表执行原理
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 现代计算机视觉入门之:什么是图片特征编码
· .NET 9 new features-C#13新的锁类型和语义
阅读排行:
· Sdcb Chats 技术博客:数据库 ID 选型的曲折之路 - 从 Guid 到自增 ID,再到
· 语音处理 开源项目 EchoSharp
· 《HelloGitHub》第 106 期
· Spring AI + Ollama 实现 deepseek-r1 的API服务和调用
· 使用 Dify + LLM 构建精确任务处理应用
点击右上角即可分享
微信分享提示