最短路之 Dijkstra 算法
普通的 Dijkstra
这是一种运用贪心的单源最短路算法,就是求从一个节点出发,到任意一个点的最短距离
首先我们要一个图

假设要求从 1 开始的单源最短路

dis[] 表示最短路数组, vis[] 表示当前节点是否被访问
那 Dijkstra 运用了贪心的思想,每次找到场上 dis 最小的且没被访问过的进行松弛操作。



进行松弛操作的节点必须是没有被访问过的

我们发现:更新 N-1 次后,剩下一个节点肯定不用更新了,所以只要 N-1 次更新。
算上找到最小值,和松弛操作,复杂度为\(O(n^2)\)。
由于代码实现简单懒,这里就不给出了。
注: Dijkstra 无法运行于负权图或求最长路(最长路贪心是错的)
优化
发现 N-1 次更新是必须的,但是找最大值可以是不是可以用数据结构优化呢?
答案是肯定的,用一个树形结构维护顶端的最小值下标
zkw 线段树优化
感谢一位 Luogu 大佬的思路!
#include<bits/stdc++.h>
#define rg register int
using namespace std;
const int inf=-1u>>1,N=100005,M=200005;
int lst[M],nxt[M],to[M],w[M],dis[N],n,m,s,fr;
namespace zkw{
int tr[N<<2],sgt=1;
inline void build(rg n){while(sgt<=n)sgt<<=1;--sgt;tr[0]=N-1;}
inline void clr(){for(rg i=1;i<=(sgt<<1)+1;i++)tr[i]=0;}
inline int cmp(const rg&x,const rg&y){return dis[x]<dis[y]?x:y;}
inline void Mdy(rg x,rg w){for(rg i=x+sgt;dis[tr[i]]>w;i>>=1)tr[i]=x;dis[x]=w;}
inline void del(rg x){tr[x+=sgt]=0;x>>=1;while(x)tr[x]=cmp(tr[x<<1],tr[x<<1|1]),x>>=1;}
}
using namespace zkw;
inline void dijkstra(rg s,rg*dis){
for(rg i=0;i<=n;i++) dis[i]=inf;clr();Mdy(s,0);
for(rg T=1;T<=n;T++){
rg u=tr[1];del(u);
for(rg i=lst[u];i;i=nxt[i])
if(dis[to[i]]>dis[u]+w[i])
Mdy(to[i],dis[u]+w[i]);
}
}
int main(){
scanf("%d%d%d",&n,&m,&s);build(n);
for(rg i=1;i<=m;i++){
scanf("%d%d%d",&fr,&to[i],&w[i]);
nxt[i]=lst[fr],lst[fr]=i;
}
dijkstra(s,dis);
for(rg i=1;i<=n;i++) printf("%d ",dis[i]);
}
或者用二叉堆优化
#include<bits/stdc++.h>
using namespace std;
const int N=100005,M=200005;
struct node {
int v,id;
node(int x,int y):v(x),id(y) {}
bool operator<(node x) const
{ return v>x.v; }
};
priority_queue<node> q;
int n,m,s,vis[N],dis[N],lst[N],nxt[M],to[M],qz[M];
inline void Dijkstra() {
memset(dis,100,sizeof(dis));
dis[s]=0;
q.push(node(0,s));
for(int u;!q.empty();) {
u=q.top().id,q.pop();
if(vis[u])continue; vis[u]=1;
for(int i=lst[u],v;i;i=nxt[i])
if(dis[v=to[i]]>dis[u]+qz[i]) {
dis[v]=dis[u]+qz[i];
q.push(node(dis[v],v));
}
}
}
int main() {
scanf("%d%d%d",&n,&m,&s);
for(int i=1,fr;i<=m;i++) {
scanf("%d%d%d",&fr,&to[i],&qz[i]);
nxt[i]=lst[fr],lst[fr]=i;
}
Dijkstra();
for(int i=1;i<=n;i++)printf("%d ",dis[i]);
}
\(P.S\) zkw 版本是本蒟蒻半年前写的,与现在的码风差别很大,
OI 一生就一次,且珍惜罢了
最短路之 Dijkstra 算法的更多相关文章
- 最短路和次短路问题,dijkstra算法
/* *题目大意: *在一个有向图中,求从s到t两个点之间的最短路和比最短路长1的次短路的条数之和; * *算法思想: *用A*求第K短路,目测会超时,直接在dijkstra算法上求次短路; ...
- 单源最短路:Dijkstra算法 及 关于负权的讨论
描述: 对于图(有向无向都适用),求某一点到其他任一点的最短路径(不能有负权边). 操作: 1. 初始化: 一个节点大小的数组dist[n] 源点的距离初始化为0,与源点直接相连的初始化为其权重,其他 ...
- HDOJ 2544 最短路(最短路径 dijkstra算法,SPFA邻接表实现,floyd算法)
最短路 Time Limit: 5000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submis ...
- [Swust OJ 842]--实验室和食堂(最短路,Dijkstra算法)
题目链接:http://acm.swust.edu.cn/problem/842/ Time limit(ms): 1000 Memory limit(kb): 10000 Description ...
- HDU - 2544最短路 (dijkstra算法)
HDU - 2544最短路 Description 在每年的校赛里,所有进入决赛的同学都会获得一件很漂亮的t-shirt.但是每当我们的工作人员把上百件的衣服从商店运回到赛场的时候,却是非常累的!所以 ...
- 最短路之Dijkstra算法
1. 邻接矩阵 int cost[MAX_V][MAX_V]; //assume cost[u][v]>0 int d[MAX_V]; bool used[MAX_V]; void Dijkst ...
- hdu2544 最短路 Dijkstra算法
最短路(Dijkstra算法模板题) Time Limit: 5000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Other ...
- 最短路模板(Dijkstra & Dijkstra算法+堆优化 & bellman_ford & 单源最短路SPFA)
关于几个的区别和联系:http://www.cnblogs.com/zswbky/p/5432353.html d.每组的第一行是三个整数T,S和D,表示有T条路,和草儿家相邻的城市的有S个(草儿家到 ...
- ACM: HDU 2544 最短路-Dijkstra算法
HDU 2544最短路 Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u Descrip ...
随机推荐
- DC-1 靶机渗透
DC-1 靶机渗透 *概况*: 下载地址 https://www.vulnhub.com/entry/dc-1,292/ *官方描述:* DC-1 is a purposely built vulne ...
- Struts2封装获取表单数据方式
一.属性封装 1.创建User实体类` package cn.entity; public class User { private String username; private String p ...
- OpenHarmony标准设备应用开发(一)——HelloWorld
(以下内容来自开发者分享,不代表 OpenHarmony 项目群工作委员会观点) 邢碌 本文是 OpenAtom OpenHarmony(以下简称"OpenHarmony")标准设 ...
- 如何在代码层面提供CPU分支预测效率
关于分支预测的基本概念和详细算法可以参考我之前写的知乎回答,基本概念不再阐述了~~ https://www.zhihu.com/question/486239354/answer/2410692045 ...
- Linux和kali Linux 介绍
常用的渗透测试平台 CTFTools kali (近亲 Ubuntu) Parrot Security OS PentestBox --由印度人开发,运行在Windows下的渗透测试环境 kali L ...
- 聊聊如何在华为云IoT平台进行产品开发
摘要:华为云物联网平台承载着南北向数据互通的功能职责. 本文分享自华为云社区<如何基于华为云IoT物联网平台进行产品开发>,作者: Super.雯 . 华为云物联网平台承载着南北向数据互通 ...
- CS基础课不完全自学指南
本文讲的是计算机学生怎么自学专业课,说长点就是该如何借助网络上已有的高质量学习资源(主要是公开课)来系统性的来点亮自己的CS技能树.这篇文章完全就是一篇自学性质的指南,需要对编程充满热情,起码觉得编程 ...
- 团队Arpha1
队名:观光队 组长博客 作业博客 组员实践情况 王耀鑫 **过去两天完成了哪些任务 ** 文字/口头描述 完成服务器连接数据库部分代码 展示GitHub当日代码/文档签入记录 接下来的计划 与服务器连 ...
- 1.17 想学好Linux,这些习惯必须养成(初学者必读)
不管是在生活还是工作中,每个人都会逐渐养成一些小习惯.坏习惯一旦形成就很难改正,所在在系统学习 Linux之前,给大家一些建议,刻意去培养一些好的习惯,对自己是很有利的. 学习Linux,要习惯使用命 ...
- 这些 Shell 分析服务器日志命令集锦,收藏好
关注「开源Linux」,选择"设为星标" 回复「学习」,有我为您特别筛选的学习资料~ 自己的小网站跑在阿里云的ECS上面,偶尔也去分析分析自己网站服务器日志,看看网站的访问量.看看 ...