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. vista忘记用户名密码的修改方法(使用PE进入系统,用cmd.exe冒充虚拟键盘,然后就可以mmc组策略,或者命令行添加用户并提升权限)

    1. 准备Windows Vista安装光盘,进入BIOS将光驱设为第一启动,在出现的安装界面依次单击"修复计算机","命令提示符". 2.输入以下命令: co ...

  2. 自定义View相关的博客收藏

    颜色: http://android.jobbole.com/83283/ 坐标: http://android.jobbole.com/83276/ 流程介绍: http://android.job ...

  3. MySQL数据库MHA+keepalive实现

    MHA(Master High Availability)目前在MySQL高可用方面是一个相对成熟的解决方案,它由日本DeNA公司youshimaton(现就职于Facebook公司)开发,是一套优秀 ...

  4. Jquery 插件开发公开属性顺序的影响.

    如下代码拷贝能正常运行. (function ($) { $.fn.DemoPlugin = function (options) { var opts; opts = $.extend({}, $. ...

  5. Delphi xe5 StyleBook的用法(待续)

    首先要在FORM里拖进来一个StyleBook1,然后在Form里设置属性,记住一定要在单击form,在OBject Inspector里设置StyleBook  [StyleBook1]. 下一个属 ...

  6. Qt常见皮肤qss代码(有Metro的风格)

    ##QTabWidget 淡蓝色效果TabWidget(属性值lightblue) QTabWidget[lightblue = "true"] QTabBar::tab{ bor ...

  7. 遗漏的SQL语句

    一.简单查询 1.限制返回行数 使用TOP n [PERCENT]选项限制返回的数据行数,TOP n说明返回n行,而TOP n PERCENT时,说明n是表示一百分数,指定返回的行数等于总行数的百分之 ...

  8. 用JavaScript刷LeetCode的正确姿势

    虽然很多人都觉得前端算法弱,但其实 JavaScript 也可以刷题啊!最近两个月断断续续刷完了 leetcode 前 200 的 middle + hard ,总结了一些刷题常用的模板代码.走过路过 ...

  9. 什么是BFC? CSS 如何使用伪元素清除浮动?

    .BFC概念: 块级格式化上下文,是一个独立的渲染区域,让处于 BFC 内部的元素与外部的元素相互隔离,使内外元素的定位不会相互影响. 我们先了解一个名词:BFC(block formatting c ...

  10. 【python3两小时根本不够】入门笔记04:线程+Lock安全同步

    有了简单爬虫,但是效率实在是太慢,于是决定启用线程进行爬取数据 但是对于临界资源的定义不好把握,思路如下: 1.定义队列(Queue的数据结构,List也可,安全性待考究) demo:https:// ...