Bellman-ford算法、SPFA算法求解最短路模板
Bellman-ford 算法适用于含有负权边的最短路求解,复杂度是O( VE ),其原理是依次对每条边进行松弛操作,重复这个操作E-1次后则一定得到最短路,如果还能继续松弛,则有负环。这是因为最长的没有环路的路,也只不过是V个点E-1条边构成的,所以松弛E-1次一定能得到最短路。因此这个算法相比 Dijkstra 首先其是对边进行增广,其次它能检测出负环的存在(若负环存在,那么最短路是取不到的,因为可以一直绕着这个负环将最小路径值不断缩小),这个弥补了 Dijkstra 的不足,但是其算法跑的比较慢,因此为了追求速度往往采用其“队列优化版”==>SPFA,因此想要理解SPFA最好先看看Bellman-ford算法。
SPFA 算法适用于含有负权边的最短路求解,其复杂度并没有网上传的那么神乎在理想情况下有论文指出其复杂度为O(kE)且k是一个约小于2的常数,但是在一些稠密图下其算法性能还是会退化到和 Bellman-ford 一样的 O( VE ),所以在稠密图下建议使用 Dij + Heap 优化的版本,稀疏图下 SPFA 还是很给力的!在 Bellman-ford 中发现啊最外层的 N-1 次循环未免盲目、实际上被松弛过的点我们希望其去继续松弛其他点,这样我们用队列将被松弛过的点存起来以便下一次继续松弛其他点,具体原理和做法可以参考下面的链接,顺便一提,SPFA还有两个优化==> SLF 与 LLL,具体也不阐述了。本文主要给出模板!
算法原理 or 学习参考链接 : 点我 、点我啦 、 点嘛!
Bellman-ford模板
- ///POJ 2387为例
- #include<bits/stdc++.h>
- using namespace std;
- ;
- const int INF = 0x3f3f3f3f;
- struct EdgeNode{ int from, to, w; };
- EdgeNode Edge[maxn*maxn];
- int Dis[maxn];
- int N, M, cnt;
- inline void init()
- {
- ; i<=N; i++)
- Dis[i] = INF;
- cnt = ;
- }
- bool BellmanFord(int st)
- {
- Dis[st] = ;
- ; i<N; i++){///N-1 次循环后肯定能找出最短路
- bool Changed = false;
- int to, from, weight;
- ; j<cnt; j++){
- to = Edge[j].to,
- from = Edge[j].from,
- weight = Edge[j].w;
- if(Dis[from]!=INF && Dis[to] > Dis[from] + weight){
- Changed = true;
- Dis[to] = Dis[from] + weight;
- ///pre[to] = j; //Record paths
- }
- }
- if(!Changed) return true;///如果没有边可以继续松弛了,说明算法结束且无负环
- if(i==N && Changed) return false;///有负环
- }
- return false; ///一般来说绝无可能执行到这一步
- }
- int main(void)
- {
- while(~scanf("%d %d", &M, &N)){
- init();
- int from, to, weight;
- ; i<M; i++){
- scanf("%d %d %d", &from, &to, &weight);
- Edge[cnt].from = from;
- Edge[cnt].to = to;
- Edge[cnt].w = weight;
- cnt++;
- Edge[cnt].to = from;
- Edge[cnt].from = to;
- Edge[cnt].w = weight;
- cnt++;
- }
- BellmanFord();
- printf("%d\n", Dis[N]);
- }
- ;
- }
SPFA模板( SLF 优化版 )
- ///POJ 2387为例
- #include <iostream>
- #include <cstdio>
- #include <cmath>
- #include <queue>
- #include <string.h>
- using namespace std;
- const int INF=0x3f3f3f3f;
- ;
- struct EdgeNode{ int v, w, nxt; };
- EdgeNode Edge[maxn*maxn];
- bool vis[maxn];
- int Head[maxn], Dis[maxn], cnt;
- int N, M;
- /// int PushCnt[maxn]; ///记录每一个节点的入队次数、方便判断负环
- inline void init()
- {
- ; i<=N; i++)
- ///PushCnt[i] = 0;
- Head[i] = -,
- Dis[i] = INF,
- vis[i] = false;
- cnt = ;
- }
- inline void AddEdge(int from, int to, int weight)
- {
- Edge[cnt].w = weight;
- Edge[cnt].v = to;
- Edge[cnt].nxt = Head[from];
- Head[from] = cnt++;
- }
- void SPFA(int st)///若要判断负环、改为 bool
- {
- deque<int> que;
- que.push_back(st);
- vis[st]=true;
- Dis[st]=;
- while (!que.empty())
- {
- int T=que.front(); que.pop_front();
- vis[T]=false;
- ; i=Edge[i].nxt)
- {
- int v=Edge[i].v;
- int w=Edge[i].w;
- if (Dis[v]>Dis[T]+w){
- Dis[v]=Dis[T]+w;
- ///p[v] = T;
- if (!vis[v]){
- ///if(++PushCnt[v] > N) return false; //有负环
- vis[v]=true;
- if(!que.empty() && Dis[v] < Dis[que.front()]) que.push_front(v);
- else que.push_back(v);
- //que.push_back(v); ///无SLF优化是这样写的
- }
- }
- }
- }
- /// return true;
- }
- int main(void)
- {
- while(~scanf("%d %d", &M, &N)){
- init();
- int from, to, weight;
- ; i<M; i++){
- scanf("%d %d %d", &from, &to, &weight);
- AddEdge(from, to, weight);
- AddEdge(to, from, weight);
- }
- SPFA();
- printf("%d\n", Dis[N]);
- }
- ;
- }
Bellman-ford算法、SPFA算法求解最短路模板的更多相关文章
- UESTC - 1987 童心未泯的帆宝和乐爷 (第k短路 A*算法+SPFA算法 模板)
传送门: http://www.qscoj.cn/#/problem/show/1987 童心未泯的帆宝和乐爷 Edit Time Limit: 10000 MS Memory Limit: ...
- 六度分离(floyd算法,SPFA算法,最短路—Dijkstra算法)
Time Limit : 5000/1000ms (Java/Other) Memory Limit : 32768/32768K (Java/Other) Total Submission(s) ...
- (模板)hdoj2544(最短路--bellman-ford算法&&spfa算法)
题目链接:https://vjudge.net/problem/HDU-2544 题意:给n个点,m条边,求点1到点n的最短路. 思路: 今天学了下bellman_ford,抄抄模板.dijkstra ...
- Bellman-Ford算法与SPFA算法详解
PS:如果您只需要Bellman-Ford/SPFA/判负环模板,请到相应的模板部分 上一篇中简单讲解了用于多源最短路的Floyd算法.本篇要介绍的则是用与单源最短路的Bellman-Ford算法和它 ...
- Bellman—Ford算法思想
---恢复内容开始--- Bellman—Ford算法能在更普遍的情况下(存在负权边)解决单源点最短路径问题.对于给定的带权(有向或无向)图G=(V,E),其源点为s,加权函数w是边集E的映射.对图G ...
- [板子]SPFA算法+链式前向星实现最短路及负权最短路
参考:https://blog.csdn.net/xunalove/article/details/70045815 有关SPFA的介绍就掠过了吧,不是很赞同一些博主说是国内某人最先提出来,Bellm ...
- 最短路模板(Dijkstra & Dijkstra算法+堆优化 & bellman_ford & 单源最短路SPFA)
关于几个的区别和联系:http://www.cnblogs.com/zswbky/p/5432353.html d.每组的第一行是三个整数T,S和D,表示有T条路,和草儿家相邻的城市的有S个(草儿家到 ...
- 用scheme语言实现SPFA算法(单源最短路)
最近自己陷入了很长时间的学习和思考之中,突然发现好久没有更新博文了,于是便想更新一篇. 这篇文章是我之前程序设计语言课作业中一段代码,用scheme语言实现单源最段路算法.当时的我,花了一整天时间,学 ...
- Bellman-Ford & SPFA 算法——求解单源点最短路径问题
Bellman-Ford算法与另一个非常著名的Dijkstra算法一样,用于求解单源点最短路径问题.Bellman-ford算法除了可求解边权均非负的问题外,还可以解决存在负权边的问题(意义是什么,好 ...
随机推荐
- 常用获取Android崩溃日志和IOS崩溃日志的几种方法
一:前言 在日常测试app时,经常会遇到崩溃问题,测试快速抓取到崩溃日志可以有效方便开发进行定位,快速解决问题所在测试做到测试分析,定位是非常重要的,这也是判断一个测试能力指标的一大维度. 二:And ...
- C++:函数求数根(总算写出来了。。。。)
[问题描述] 数根问题递归求解:输入n个正整数(输入格式中第一行为整数个数n,后续行为n个整数),输出各个数的数根.数根的定义:对于一个正整数n,我们将它的各个位相加得到一个新的数字,如果这个数字是一 ...
- C#的Split()方法
var arr = list[i]["Tag"].Split(new char[] { ',' }, StringSplitOptions.RemoveEmptyEntries);
- 使用原生js 获取用户访问项目的浏览器类型
想要获取浏览器的类型很简单,网上提供了很多方法,但是看过之后,都是根据浏览器内核来判断是ie,谷歌,火狐,opeara的, 所以不能进一步判断在国内使用的主流浏览器类型,比如360,百度,搜狐浏览器等 ...
- SVN服务器和客户端使用教程总结
一.SVN简介 Subversion是什么? 它是一个自由/开源的版本控制系统,一组文件存放在中心版本库,记录每一次文件和目录的修改,Subversion允许把数据恢复到早期版本,或是检查数据修改的历 ...
- Python实现批量执行华为交换机脚本
#!/usr/bin/python3 # -*- coding:utf-8 -*- import paramiko import time ssh = paramiko.SSHClient() key ...
- freemarker进阶--项目中使用
1.工程引入依赖 <dependency> <groupId>org.freemarker</groupId> <artifactId>freemark ...
- 05: 常用反扒机制 & 解决方法
1.1 常用反扒机制 参考博客:https://blog.csdn.net/python36/article/details/90174300 1.header 浏览器的请求头 header中添 ...
- mysql简单命令
库: 增 create database db1:新建一个默认编码的库 create database db1 charset uet8 ;建一个编码为 utf8 的库 删 drop database ...
- Diango路由映射FBV和CBV
django中请求处理方式有2种:FBV(function base views) 和 CBV(class base views),换言之就是一种用函数处理请求,一种用类处理请求. FBV # url ...