Johnson 算法

全源最短路径求解其实是单源最短路径的推广,求解单源最短路径的两种算法时间复杂度分别为:

  • Dijkstra 单源最短路径算法:时间复杂度为 \(O(E + VlogV)\),要求权值非负;
  • Bellman-Ford 单源最短路径算法:时间复杂度为 \(O(VE)\),适用于带负权值情况;

如果对全图顶点遍历,使用 Dijkstra 算法,时间复杂度将变成 \(O(VE + V2logV)\),看起来优于 Floyd-Warshall 算法的 \(O(V3)\)。不过,Dijkstra 算法要求权值重不能为负。

Johnson 算法能调整权重为负的图,使之能够使用 Dijkstra 算法。

以下图为例,Johnson 算法对下图进行re-weight操作,使权重不为负,并且re-weight后,计算出来的最短路径仍然正确。

首先,新增一个源顶点 ,并使其与所有顶点连通,新边赋权值为 0,如下图所示。

接下来重新计算新增顶点到其它顶点的最短路径,利用单源最短路径算法,图中存在负权重节点,使用bellman ford算法,计算新增节点到其它节点的最短路径 h[],然后使用如下公式对所有边的权值进行 "re-weight":

w(u, v) = w(u, v) + (h[u] - h[v]).

对于此公式的证明请参考算法导论一书。

现在除新增结点外,其它结点的相关边权重值都已经为正数了,可以将新增结点删除,对其它结点使用Dijkstra 算法了。

例题

【模板】Johnson 全源最短路

#include<bits/stdc++.h>
using namespace std;
#define int long long
int read(){
int x=0;bool f=0;char ch=getchar();
while(!isdigit(ch))f|=ch=='-',ch=getchar();
while(isdigit(ch))x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
return f?-x:x;
} void write(int x){
if(x<0)putchar('-'),x=-x;
if(x>9)write(x/10);
putchar(48+x%10);
} void writeln(int x){write(x);putchar('\n');}
void writebl(int x){write(x);putchar(' ');} #define I inline
#define R register const int maxn = 3e3+5;
const int maxm = maxn*6;
#define inf 1000000000 struct Johnson{
struct edge{int v,w,next;}e[maxm];
int head[maxn],vis[maxn],dis[maxn],tot,h[maxn];
void add(int u,int v,int w){e[++tot]=(edge){v,w,head[u]};head[u]=tot;}
struct node{
int dis;int pos;
bool operator < (const node &x)const{return x.dis<dis;}
};
priority_queue<node> q;
I void dijkstra(int s){
memset(dis,0x3f,sizeof(dis));memset(vis,0,sizeof(vis));
dis[s] = 0;
q.push((node){0,s});
while(!q.empty()){
node tmp = q.top();q.pop();
int x = tmp.pos,d = tmp.dis;
if(vis[x])continue;vis[x] = 1;
for(int i = head[x];i;i = e[i].next){
int v = e[i].v;
if(dis[v] > dis[x] + e[i].w){
dis[v] = dis[x] + e[i].w;
if(!vis[v])q.push((node){dis[v],v});
}
}
}
}
int tim[maxn];
I bool spfa(int s,int n){
queue<int> q;
memset(h,0x3f,sizeof(h));memset(vis,0,sizeof(vis));
h[s]=0;vis[s]=1;q.push(s);
while(!q.empty()){
int u=q.front();q.pop();vis[u]=0;
if(++tim[u]>n-1)return 0;
for(R int i=head[u];i;i=e[i].next){
int v=e[i].v;
if(h[v]>h[u]+e[i].w){
h[v]=h[u]+e[i].w;
if(!vis[v])q.push(v),vis[v]=1;
}
}
}
return 1;
}
}J;
int n,m;
signed main(){
n=read(),m=read();
for(R int i=1,u,v,w;i<=m;++i){
u=read(),v=read(),w=read();
J.add(u,v,w);
}
for(R int i=1;i<=n;++i)J.add(0,i,0);
if(!J.spfa(0,n)){puts("-1");return 0;}
for(R int u=1;u<=n;++u)
for(R int i=J.head[u];i;i=J.e[i].next){
int v=J.e[i].v;
J.e[i].w+=J.h[u]-J.h[v];
}
for(R int i=1;i<=n;++i){
J.dijkstra(i);
long long ans=0;
for(R int j=1;j<=n;++j){
if(J.dis[j]==J.dis[n+1])ans+=j*inf;
else ans+=j*(J.dis[j]+J.h[j]-J.h[i]);
}
writeln(ans);
}
}

Johnson 最短路算法的更多相关文章

  1. Johnson算法:多源最短路算法

    Johnson算法 请不要轻易点击标题 一个可以在有负边的图上使用的多源最短路算法 时间复杂度\(O(n \cdot m \cdot log \ m+n \cdot m)\) 空间复杂度\(O(n+m ...

  2. Dijkstra 最短路算法(只能计算出一条最短路径,所有路径用dfs)

    上周我们介绍了神奇的只有五行的 Floyd 最短路算法,它可以方便的求得任意两点的最短路径,这称为"多源最短路".本周来来介绍指定一个点(源点)到其余各个顶点的最短路径,也叫做&q ...

  3. Dijkstra最短路算法

    Dijkstra最短路算法 --转自啊哈磊[坐在马桶上看算法]算法7:Dijkstra最短路算法 上节我们介绍了神奇的只有五行的Floyd最短路算法,它可以方便的求得任意两点的最短路径,这称为“多源最 ...

  4. Floyd最短路算法

    Floyd最短路算法 ----转自啊哈磊[坐在马桶上看算法]算法6:只有五行的Floyd最短路算法 暑假,小哼准备去一些城市旅游.有些城市之间有公路,有些城市之间则没有,如下图.为了节省经费以及方便计 ...

  5. Book 最短路算法

    用HDU2544整理一下最近学的最短路算法 1.Dijkstra算法 原理:集合S表示已经找到最短路径的点,d[]表示当前各点到源点的距离 初始时,集合里面只有源点,当每个点u进入集合S时,用d[u] ...

  6. 近十年one-to-one最短路算法研究整理【转】

    前言:针对单源最短路算法,目前最经典的思路即标号算法,以Dijkstra算法和Bellman-Ford算法为根本演进了各种优化技术和算法.针对复杂网络,传统的优化思路是在数据结构和双向搜索上做文章,或 ...

  7. 【啊哈!算法】算法7:Dijkstra最短路算法

    上周我们介绍了神奇的只有五行的Floyd最短路算法,它可以方便的求得任意两点的最短路径,这称为“多源最短路”.本周来来介绍指定一个点(源点)到其余各个顶点的最短路径,也叫做“单源最短路径”.例如求下图 ...

  8. 【啊哈!算法】算法6:只有五行的Floyd最短路算法

            暑假,小哼准备去一些城市旅游.有些城市之间有公路,有些城市之间则没有,如下图.为了节省经费以及方便计划旅程,小哼希望在出发之前知道任意两个城市之前的最短路程.         上图中有 ...

  9. Comet OJ 热身赛(E题)(处理+最短路算法)

    dijkstra 已经提交 已经通过 42.86% Total Submission:189 Total Accepted:81 题目描述 Eagle Jump公司正在开发一款新的游戏.泷本一二三作为 ...

  10. 【最短路算法】Dijkstra+heap和SPFA的区别

    单源最短路问题(SSSP)常用的算法有Dijkstra,Bellman-Ford,这两个算法进行优化,就有了Dijkstra+heap.SPFA(Shortest Path Faster Algori ...

随机推荐

  1. VMware 备份操作系统

    在VMware 中备份方式有两种:快照和克隆. 快照:又称还原点,就是保存在拍快照时系统的状态,包含所有内容.在之后的使用中,随时都可以恢复.[短期备份,需要频繁备份时,使用该方法.操作的虚拟系统一般 ...

  2. 是时候丢掉BeanUtils了

    前言 为了更好的进行开发和维护,我们都会对程序进行分层设计,例如常见的三层,四层,每层各司其职,相互配合.也随着分层,出现了VO,BO,PO,DTO,每层都会处理自己的数据对象,然后向上传递,这就避免 ...

  3. python2.7源码安装方式

    安装python2.7 下载Python 2.7, 下载地址 解压安装 tar -xzvf Python-2.7.15.tgz cd Python-2.7.15 ./configure --prefi ...

  4. cdn 引入的资源需要通过 externals 排除打包哦~

    cdn 指的是通过相互连接的网络系统,使用最靠近用户的服务器将音乐.图片等资源以高效率和低成本的方式将内容传递给用户. 在 webpack 中,我们可能会将引入的第三方资源会编译成单独的文件,作为静态 ...

  5. Django+anaconda(spyder)

    一.搭建django虚拟环境 打开anaconda prompt 输入:conda create -n mydjango_env 判断(y/n):y 查看虚拟环境 conda env list *号表 ...

  6. 【NestJS系列】核心概念:Module模块

    theme: fancy highlight: atelier-dune-dark 前言 模块指的是使用@Module装饰器修饰的类,每个应用程序至少有一个模块,即根模块.根模块是Nest用于构建应用 ...

  7. Kafka Stream 处理器API

    6.1章节内容 了解如何使用处理器API对以下场景进行处理 ①以有规律的间隔定期执行 ②将控制记录如何向下游发送 ③将记录转发给特定的子节点 ④创建Kafka Streams API中不存在的功能 6 ...

  8. MutationObserver监听dom元素结构及属性变化

    工作中埋码需求,当某些动态插入的元素出现时触发埋码事件,因此需要对插入元素的父节点进行监听,子节点发生变化时触发相应埋码逻辑. 方法一 监听页面结构及子元素变化: (function () { //事 ...

  9. FastGPT 接入飞书(不用写一行代码)

    FastGPT V4 版本已经发布,可以通过 Flow 可视化进行工作流编排,从而实现复杂的问答场景,例如联网谷歌搜索,操作数据库等等,功能非常强大,还没用过的同学赶紧去试试吧. 飞书相比同类产品算是 ...

  10. Solution Set -「ARC 124」

    「ARC 124A」LR Constraints Link. 我们可以把 \(1\sim n\) 个盒子里能放的球的编号集合全部求出来.然后就直接来. 注意题目已经给出了 \(k\) 个球的位置,所以 ...