dijkstra是一种单源最短路径算法,即求一个点到其他点的最短路。不能处理负边权。

最近某种广为人知的算法频繁被卡,让dijkstra逐渐成为了主流,甚至在初赛中鞭尸了SPFA(?

dijkstra效率还是不错的,而且不容易被卡。

一、主要思想

用dis数组保存最短路径。初始化时,dis[s]设置为0(s为起点),其他无穷大。

枚举点i,每次从i能够到达的点中,选出权值最小的一个mpl(min place 个人叫法),把dis[mpl]直接设置为权值。可以证明,这就是最短的路径。

证明:若不是,则必有点k,能使

s->k+k->mpl < s->mpl

由权值最小得 s->mpl < s->k 明显矛盾

(我居然没写易证!

再枚举mpl到达的点k,若mpl->k+dis[mpl] > dis[k] 则更新dis[k]。专业术语叫做:松弛。

以此类推,更新完所有的点,算法结束。复杂度O(n2)

代码:

#include<iostream>
#include<vector>
#include<cstring> using namespace std; //GDT TQL Orz struct edge {
int next,to,dis;
} g[];//前向星 int dis[],gdt[],n,m,s,u,v,w,ne=;
//gdt就是head数组。此处为玩梗QwQ inline void add(int from,int to,int d) {
g[++ne].next=gdt[from];
g[ne].to=to;
g[ne].dis=d;
gdt[from]=ne;
}//前向星建边 inline void dijkstra() {
int mn,mpl=s;
bool vis[]= {};
dis[s]=;
for(register int i=;i<=n;i++)
{
mn=;
for(register int now=; now<=n; now++) {
if(dis[now]<mn&&!vis[now]) {
mn=dis[now];
mpl=now;
}
}
//选出权值最小的点
vis[mpl]=;
//此节点已被拓展,也就是变为白点(你们可能会从奇奇怪怪
//的算法书中得到这种说法)。
for(register int now=gdt[mpl]; now; now=g[now].next) {
if(!vis[g[now].to]&&dis[g[now].to]>dis[mpl]+g[now].dis)
dis[g[now].to]=dis[mpl]+g[now].dis;
} }
} int main() {
memset(gdt,,sizeof(gdt));
memset(g,,sizeof(g));
cin>>n>>m>>s;
for(int i=; i<=n; i++)
dis[i]=; for(int i=; i<=m; i++) {
cin>>u>>v>>w;
add(u,v,w);
}
dijkstra();
for(int i=; i<=n; i++) {
cout<<dis[i]<<' ';
}
return ;
}

二、堆优化

所谓堆,实际相当于一个优先队列。

每次把点入队,使用时直接可以访问离s距离最小的。

具体实现时,会用STL中的priority_queue(优先队列)。而且,一个点需要一个地址(访问)和一个距离s的最短距离(比较、更新最短路径)。

代码:

#include<iostream>
#include<vector>
#include<cstring>
#include<queue> using namespace std; //GDT TQL Orz struct edge
{
int next,to,dis;
}g[]; struct node
{
int k,v;
//地址key,权值value
}; bool operator < (node n1,node n2)
{
return n1.v>n2.v;
} //优先队列维护时需要比较,而结构体无法直接比较,使用重载运算符。 int dis[],gdt[],n,m,s,u,v,w,ne=; inline void add(int from,int to,int d)
{
g[++ne].next=gdt[from];
g[ne].to=to;
g[ne].dis=d;
gdt[from]=ne;
} inline void dijkstra()
{
int mn,mpl=s;
node nd,temp;
bool vis[]= {};
dis[s]=;
priority_queue<node> pq;
temp.k=s;
temp.v=;
pq.push(temp);//第一个点更新起点
while(!pq.empty())
{
nd=pq.top();
pq.pop(); if(vis[nd.k]) continue;
vis[nd.k]=;//拓展
mpl=nd.k;
for(register int now=gdt[mpl]; now; now=g[now].next)
{
if(!vis[g[now].to]&&dis[g[now].to]>dis[mpl]+g[now].dis)
{
dis[g[now].to]=dis[mpl]+g[now].dis;
temp.k=g[now].to;
temp.v=dis[g[now].to];
pq.push(temp);//新的一个结点
}
}
}
} int main() {
memset(gdt,,sizeof(gdt));
memset(g,,sizeof(g));
cin>>n>>m>>s;
for(int i=; i<=n; i++)
dis[i]=; for(int i=; i<=m; i++)
{
cin>>u>>v>>w;
add(u,v,w);
}
dijkstra();
for(int i=; i<=n; i++)
{
cout<<dis[i]<<' ';
}
return ;
}

三、关于SPFA

如果一个图没有负边权,那就一定会卡你的SPFA。

dijkstra算法学习笔记的更多相关文章

  1. Johnson算法学习笔记

    \(Johnson\)算法学习笔记. 在最短路的学习中,我们曾学习了三种最短路的算法,\(Bellman-Ford\)算法及其队列优化\(SPFA\)算法,\(Dijkstra\)算法.这些算法可以快 ...

  2. Johnson 全源最短路径算法学习笔记

    Johnson 全源最短路径算法学习笔记 如果你希望得到带互动的极简文字体验,请点这里 我们来学习johnson Johnson 算法是一种在边加权有向图中找到所有顶点对之间最短路径的方法.它允许一些 ...

  3. C / C++算法学习笔记(8)-SHELL排序

    原始地址:C / C++算法学习笔记(8)-SHELL排序 基本思想 先取一个小于n的整数d1作为第一个增量(gap),把文件的全部记录分成d1个组.所有距离为dl的倍数的记录放在同一个组中.先在各组 ...

  4. Manacher算法学习笔记 | LeetCode#5

    Manacher算法学习笔记 DECLARATION 引用来源:https://www.cnblogs.com/grandyang/p/4475985.html CONTENT 用途:寻找一个字符串的 ...

  5. dijkstra算法学习

    dijkstra算法学习 一.最短路径 单源最短路径:计算源点到其他各顶点的最短路径的长度 全局最短路径:图中任意两点的最短路径 Dijkstra.Bellman-Ford.SPFA求单源最短路径 F ...

  6. 某科学的PID算法学习笔记

    最近,在某社团的要求下,自学了PID算法.学完后,深切地感受到PID算法之强大.PID算法应用广泛,比如加热器.平衡车.无人机等等,是自动控制理论中比较容易理解但十分重要的算法. 下面是博主学习过程中 ...

  7. 算法学习笔记(三) 最短路 Dijkstra 和 Floyd 算法

    图论中一个经典问题就是求最短路.最为基础和最为经典的算法莫过于 Dijkstra 和 Floyd 算法,一个是贪心算法,一个是动态规划.这也是算法中的两大经典代表.用一个简单图在纸上一步一步演算,也是 ...

  8. SPFA算法学习笔记

    一.理论准备 为了学习网络流,先水一道spfa. SPFA算法是1994年西南交通大学段凡丁提出,只要最短路径存在,SPFA算法必定能求出最小值,SPFA对Bellman-Ford算法优化的关键之处在 ...

  9. 算法学习笔记——sort 和 qsort 提供的快速排序

    这里存放的是笔者在学习算法和数据结构时相关的学习笔记,记录了笔者通过网络和书籍资料中学习到的知识点和技巧,在供自己学习和反思的同时为有需要的人提供一定的思路和帮助. 从排序开始 基本的排序算法包括冒泡 ...

随机推荐

  1. 反射:获取枚举类型的Name,Value,Description

    [Obsolete("请使用新的方法XXX")] //使用Obsolete特性来告诉使用者这是一个过期的方法 private static void Test() { Type t ...

  2. RPG Maker MV游戏解包

    该文章最新版本请前往:https://www.crowsong.xyz/127.html 前言 使用Petschko's RPG-Maker-MV File-Decrypter进行解包 使用Petsc ...

  3. Qt使Release版本可调试

    只需在pro文件中加入 QMAKE_CXXFLAGS_RELEASE = $$QMAKE_CFLAGS_RELEASE_WITH_DEBUGINFO QMAKE_LFLAGS_RELEASE = $$ ...

  4. Qt4学习笔记 (7) 本篇说一下Qt对于脚本的支持, 即QtScript模块.

    本篇说一下Qt对于脚本的支持, 即QtScript模块. Qt支持的脚本基于ECMAScript脚本语言, 这个东西又是javascript, jscript的基础. 所以, 一般只要学过javasc ...

  5. 递归导致的StackOverflow的分析

    递归在多层次遍历时尤为重要,这里我们不讲递归的实现,来谈谈递归的内存占用情况. 如下代码,当我们运行时很简单,StackOverflowException瞬间抛出:这里确实是“瞬间”出错了,线程堆栈溢 ...

  6. SYN2136型 北斗NTP网络时间服务器

    SYN2136型  北斗NTP网络时间服务器 北斗NTP网络时间服务器时间服务器使用说明视频链接: http://www.syn029.com/h-pd-109-0_310_36_-1.html 请将 ...

  7. 仿写confirm和alert弹框

    在工作中,我们常常会遇到原生的样式感觉比较丑,又和我们做的项目风格不搭.于是就有了仿写原生一些组件的念头,今天我就带大家仿写一下confirm和alert样式都可以自己修改. 有些的不好的地方请指出来 ...

  8. 容器化之Docker小知识普及

    Docker确实是个好东西,相当于一个小型虚拟机,里面环境隔离只要有对应的镜像就能运行业务应用.运行基础应用.能跑数据库等等,还能跑linux等等. Docker 从狭义上来讲就是一个进程,从广义上来 ...

  9. 5个现在就该使用的数组Array方法: indexOf/filter/forEach/map/reduce详解(转)

    ECMAScript5标准发布于2009年12月3日,它带来了一些新的,改善现有的Array数组操作的方法.然而,这些新奇的数组方法并没有真正流行起来的,因为当时市场上缺乏支持ES5的浏览器.     ...

  10. C#常用设计模式--单例模式

    为什么要使用单例模式 在我们的整个游戏生命周期当中,有很多对象从始至终有且只有一个.这个唯一的实例只需要生成一次,并且直到游戏结束才需要销毁.  单例模式一般应用于管理器类,或者是一些需要持久化存在的 ...