图的最短路算法 Bellman-Ford
BF求图的最短路径的时间复杂度是O(MN),这样的时间复杂度并不比迪杰斯特拉算法好,但是BF算法支持图中存在负权的情况,但图中不能存在负圈,因为如果存在负圈,最短路是不存在的,因此BF算法的另一个重要应用是判负圈(如果松弛了N-1次后,还能松弛,就说明存在负圈)
简单写法(没有判断是否存在负圈。判断负圈只要在最后判断能否继续进行松弛即可)
#include<iostream>
using namespace std;
;
<<;
int d[maxn],w[maxn],f[maxn],t[maxn];
int main()
{
int N,M;
cin>>N>>M;
;i<M;i++){
cin>>f[i]>>t[i]>>w[i];
}
;i<=N;i++) d[i]=inf;
d[]=;
;i<=N-;i++){//最短路径最多经过N-1个节点 需要N-1轮松弛
;j<M;j++){//枚举所有的边
int x=f[j],y=t[j];
if(d[x]<inf) d[y]=min(d[y],d[x]+w[j]);
}
}
cout<<d[N];
;
}
可以用队列进行优化,代替上面的循环检查(代码来自 刘汝佳《算法竞赛入门经典》)
#include<iostream>
#include<vector>
#include<cstring>
#include<queue>
using namespace std;
<<;
;
struct Edge{
int from,to,dist;
};
vector<Edge> edges;
vector<int> G[maxn];
int N,M;
int inq[maxn]/*在队列里的标记*/,cnt[maxn]/*某个节点进入队列的次数,也就是松弛的次数*/;
int d[maxn];
bool bellman_ford(int s){
queue<int> Q;
memset(inq,,sizeof(inq));
memset(cnt,,sizeof(cnt));
;i<=N;i++) d[i]=inf;
d[s]=;inq[s]=;Q.push(s);
while(!Q.empty()){
int u=Q.front();Q.pop();
inq[u]=;//出队
;i<G[u].size();i++){
Edge& e=edges[G[u][i]];
if(d[u]<inf&&d[e.to]>d[u]+e.dist){
d[e.to]=d[u]+e.dist;
if(!inq[e.to]){
Q.push(e.to);inq[e.to]=;//入队
if(++cnt[e.to]>N) return false;
}
}
}
}
return true;
}
int main()
{
int f,t,dd;
cin>>N>>M;
;i<M;i++){
cin>>f>>t>>dd;
edges.push_back((Edge){f,t,dd});
int m=edges.size();
G[f].push_back(m-);
}
))//起点是1
cout<<d[N];//终点是N
else cout<<"Impossible!";//存在负圈
;
}
图的最短路算法 Bellman-Ford的更多相关文章
- 图的最短路算法 Dijkstra及其优化
单源最短路径算法 时间复杂度O(N2) 优化后时间复杂度为O(MlogN)(M为图中的边数 所以对于稀疏图来说优化后更快) 不支持有负权的图 #include<iostream> usin ...
- 图的最短路算法 Floyd
多源最短路径算法 时间复杂度O(N3) 简单修改可求有向图的传递闭包 #include<iostream> using namespace std; const int maxn=1024 ...
- Bellman - Ford 算法解决最短路径问题
Bellman - Ford 算法: 一:基本算法 对于单源最短路径问题,上一篇文章中介绍了 Dijkstra 算法,但是由于 Dijkstra 算法局限于解决非负权的最短路径问题,对于带负权的图就力 ...
- poj1860 bellman—ford队列优化 Currency Exchange
Currency Exchange Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 22123 Accepted: 799 ...
- Bellman—Ford算法思想
---恢复内容开始--- Bellman—Ford算法能在更普遍的情况下(存在负权边)解决单源点最短路径问题.对于给定的带权(有向或无向)图G=(V,E),其源点为s,加权函数w是边集E的映射.对图G ...
- 算法专题 | 10行代码实现的最短路算法——Bellman-ford与SPFA
今天是算法数据结构专题的第33篇文章,我们一起来聊聊最短路问题. 最短路问题也属于图论算法之一,解决的是在一张有向图当中点与点之间的最短距离问题.最短路算法有很多,比较常用的有bellman-ford ...
- Til the Cows Come Home 最短路Dijkstra+bellman(普通+优化)
Til the Cows Come Home 最短路Dijkstra+bellman(普通+优化) 贝西在田里,想在农夫约翰叫醒她早上挤奶之前回到谷仓尽可能多地睡一觉.贝西需要她的美梦,所以她想尽快回 ...
- ACM/ICPC 之 最短路径-Bellman Ford范例(POJ1556-POJ2240)
两道Bellman Ford解最短路的范例,Bellman Ford只是一种最短路的方法,两道都可以用dijkstra, SPFA做. Bellman Ford解法是将每条边遍历一次,遍历一次所有边可 ...
- Dijkstra 最短路算法(只能计算出一条最短路径,所有路径用dfs)
上周我们介绍了神奇的只有五行的 Floyd 最短路算法,它可以方便的求得任意两点的最短路径,这称为"多源最短路".本周来来介绍指定一个点(源点)到其余各个顶点的最短路径,也叫做&q ...
随机推荐
- Oracle-decode函数
decode函数 简单例子:管理员登录Oracle select sid, username, decode(command, 0, 'None', 2, 'Insert', 3, 'Select', ...
- onclick事件对动态参数类型为字符串的处理
onclick="solveRow("'+row.isbesolve+'")"
- java.util.concurrent
软件包 java.util.concurrent 的描述 在并发编程中很常用的实用工具类.此包包括了几个小的.已标准化的可扩展框架,以及一些提供有用功能的类,没有这些类,这些功能会很难实现或实现起来冗 ...
- 0814JavaScript简介、基本语法、运算符、转换
一.JavaScript简介 1.JavaScript是个什么东西? 它是个脚本语言,需要有宿主文件,它的宿主文件是HTML文件. 2.它与Java什么关系? 没有什么直接的联系,Java是Sun公司 ...
- ylbtech-Unitity-CS:Hello world
ylbtech-Unitity-CS:Hello world 1.A,效果图返回顶部 1.B,源代码返回顶部 1.B.1,Hello1.cs public class Hello1 { publi ...
- unset是不能清除保存在本地电脑上的cookie的,用于session就可以(弄了半天原来是这样)
unset($_COOKIE["historyWord[$wordId]"]); 这样是不行的,unset只是将变量在脚本运行时注销,但是cookie是写在客户端的,下一次还是可以 ...
- OOP三个基本特征:封装、继承、多态
面向对象的三个基本特征是:封装.继承.多态. 封装 封装最好理解了.封装是面向对象的特征之一,是对象和类概念的主要特性. 封装,也就是把客观事物封装成抽象的类,并且类可以把自己的数据和方法只让可信的类 ...
- git的忽略文件和删除文件操作
1 删除工作区和暂存去的a文件$ git rm a 2只删除暂存去的 a文件,a文件就不被跟踪了.可以执行git add a从新添加回暂存去$ git rm --cached a 3 git mv 操 ...
- js判断正整数
var r = /^\+?[1-9][0-9]*$/; //判断正整数 r.test(str);
- [技巧]把Excel里的数据插入到数据库里的方法
.如果先在6行数据的最后一列在插入一列数据,请先把列名写好,然后再第一行的该列下输入数字, 然后选中该单元格向下拖拽一个单元格然后就能看到黑色的小框, 双击右下角黑色的小点,6行数据就 会填上默认的第 ...