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. Windows 10 (IIS 10)安装Microsoft Web Farm Framework Version 2.2 for IIS7问题

    But I got an error message "iis version 7.0 or greater is required to install Web Farm Framewor ...

  2. fastjson 出现首字母小写的问题

    今天工作使用fastjson要求传过去的参数全为大写,在使用的过程中发现它自动将我的字段首字母转为小写了,在网上查了一些资料,发现下面的这个挺好,比其他的要方便. package com.alibab ...

  3. storm和kafka的wordCount

    这个是在window环境下面安装的kafka 下载pom依赖 <dependency> <groupId>org.apache.storm</groupId> &l ...

  4. 迁移到MSYS2 与 Qt 工具链注意的几个事情(g++在链接时,符号依赖项查找遵循从左至右的顺序,但qmake会自动合并造成错误。使用脚本给Mingw32-make创造出一个局部的VC编译环境)

    Microsoft Visual Studio 2015社区版提供了强大的开发体验,且 Qt 提供了预编译版本.然而,由于客户提出兼容Windows XP ~ Windows 8.1 这样宽泛的环境要 ...

  5. InnoExtractor可以解压一些安装文件,以获得其中的特殊文件

    you can use InnoExtractor unpack the installer file and get uniSimpleEnc.dcu file. https://www.board ...

  6. delphi中pos和Ansipos函数的区别

    POS和ANSIPOS的主要区别在于,POS参数多用于是一个字符,而ANSIPOS多用于是多个字符: 例如:i:=pos('a','sssssas'); i:=ansipos('abc','sdabc ...

  7. 怎样让窗口不显示在任务栏和ALT+TAB中(隐藏窗口再嵌套,几乎是万能的办法)

    之前想弄个像QQ旋风那样的悬浮窗口,就研究了下怎么让窗口不显示在任务栏中,方法其实很简单就是将窗口的扩张属性设置成WS_EX_TOOLWINDOW,MSDN中对该属性有详细介绍,代码如下: ::Set ...

  8. WSAAsyncSelect Demo

    #include <WinSock2.h> #include <Windows.h> #include <StrSafe.h> #pragma comment(li ...

  9. TCP使用注意事项总结

    目录 发送或者接受数据过程中对端可能发生的情况汇总 本端TCP发送数据时对端进程已经崩溃 本端TCP发送数据时对端主机已经崩溃 本端TCP发送数据时对端主机已经关机 某个连接长时间没有数据流动 TCP ...

  10. pytorch实现yolov3(2) 配置文件解析及各layer生成

    配置文件 配置文件yolov3.cfg定义了网络的结构 .... [convolutional] batch_normalize=1 filters=64 size=3 stride=2 pad=1 ...