Dijkstra算法

由于我之前一直记的迪杰斯特拉的翻译导致我把dijkstra写成了dijstra……所以下文#define dijstra dijkstra

我以后叫她迪杰克斯歘!

Dijskra是用来在有向图或者无向图中寻找任意两个点的最小距离的算法。它相较于spfa不会死掉(spfa死了),但是无法处理带负环的图和求最长路(除非加上一些奇怪的东西,我们这里是新手村不予讨论我不会)

Dijskra的核心思想是先把所有边的边权进行从小到大的排序,再建一个记录起点到各个点的距离的dis数组并初始化其为一个最大值(inf=!0U>>1,或者0x3f3f3f3f什么的,但不要自己觉得很大但是小了。注意,要把到起点的距离dis[start]设为0),每次讨论一个点i遍历所有和它相连的边到的点j,如果起点到i的距离加上边权小于顶点到j的距离,那么更新j的距离。然后再选一个最近的点标记一下vis,再次进行遍历。处理完之后得到dis[end]就是起点到终点的距离了。

至于dijskra的推导和证明,我不会我们在这里不予讨论。还是不懂的话可以搜一搜其他入门帖子(有图的那种)

存图的话,可以用邻接矩阵和前向星。

Dijskra的堆优化

我们发现,每次遍历一个点就需要把所有的边都扫一遍然后进行松弛。这个导致朴素dijskra非常慢。所以我们可以想个法子优化一下。就是——使用小根堆。我们可以发扬C++的伟大之处,用priority_queue和pair进行优化。

讲解一下pair关键字。它在C++的<utility>(发音[juˈtɪlətɪ])头文件中。简单来说就是一个只有两个元素的结构体。加入我们定义了一个pair <int,int>p;的pair,那么我们可以用p.first和p.second来分别调用他的两个元素。pair有什么好处呢?当我们使用sort的时候,它会先根据第一关键字进行排列,当第一关键词相同时再根据第二关键字排列。所以我们用pair存边的时候,用first存边权,用second存编号即可。

make_pair(a,b)关键字可以把a和b弄成一个pair。这在我们要把一个pair放进优先队列时有用。

我们再来看看优先队列的部分。裸的优先队列是大根堆(从大到小),我们需要把它重新定义一下变成小根堆,即:

priority_queue<int,vector<int>,greater<int> >q;

greater操作符就是重载运算符()变成了(>)。less和它相反。它们在头文件<functional>里。记得引用。(似乎algorithm已经包含了)

所以我们就可以轻松得到优化后的代码~

typedef pair<int,int> pii;
int dis[maxn];
bool vis[maxn];
inline void dijkstra(int start,int end)
{ memset(dis,0x3f3f3f3f,sizeof dis);
priority_queue<pii,vector<pii >,greater<pii > > q;
q.push(make_pair(0,start)),dis[start]=0;
while(!q.empty())
{
int u=q.top().second;q.pop();
if(vis[u])continue;
vis[u]=1;
for(int i=head[u];i;i=nxt[i])
{
int v=t[i],w=val[i];
if(dis[v]>=dis[u]+w)
{
dis[v]=dis[u]+w;
q.push(make_pair(dis[v],v));
}
}
}
}

(update:之前没有加入vis数组是因为我认为它没有影响,抱歉。)

Dijkstra和堆优化的更多相关文章

  1. 最短路模板(Dijkstra & Dijkstra算法+堆优化 & bellman_ford & 单源最短路SPFA)

    关于几个的区别和联系:http://www.cnblogs.com/zswbky/p/5432353.html d.每组的第一行是三个整数T,S和D,表示有T条路,和草儿家相邻的城市的有S个(草儿家到 ...

  2. Dijkstra及其堆优化

    朴素Dijkstra #include<bits/stdc++.h> using namespace std; const int inf=9999999; bool book[105]; ...

  3. 洛谷 P4779 【dijkstra】+(堆优化)+(链式前向星) (模板题)

    <题目链接> 题目描述 给定一个 N 个点, M 条有向边的带非负权图,请你计算从 S 出发,到每个点的距离. 数据保证你能从 S 出发到任意点. 输入格式: 第一行为三个正整数 N,M, ...

  4. POJ 3635 - Full Tank? - [最短路变形][手写二叉堆优化Dijkstra][配对堆优化Dijkstra]

    题目链接:http://poj.org/problem?id=3635 题意题解等均参考:POJ 3635 - Full Tank? - [最短路变形][优先队列优化Dijkstra]. 一些口胡: ...

  5. Dijkstra+优先队列 堆优化

    关于堆优化 传统\(Dijkstra\),在选取中转站时,是遍历取当前最小距离节点,但是我们其实可以利用小根堆(STL的priority_queue)优化这个过程,从而大大降低复杂(\(O(V2+E) ...

  6. Dijkstra的堆优化

    先附上一个例题:P3371 [模板]单源最短路径 一眼扫去,最短路... spfa可行,但是今天的主题是Dijkstra: #include<iostream> #include<a ...

  7. Dijkstra算法堆优化

    转自 https://blog.csdn.net/qq_41754350/article/details/83210517 再求单源最短路径时,算法有优劣之分,个人认为在时间方面 朴素dijkstra ...

  8. Dijkstra算法堆优化详解

    DIJ算法的堆优化 DIJ算法的时间复杂度是\(O(n^2)\)的,在一些题目中,这个复杂度显然不满足要求.所以我们需要继续探讨DIJ算法的优化方式. 堆优化的原理 堆优化,顾名思义,就是用堆进行优化 ...

  9. 最短路-朴素版Dijkstra算法&堆优化版的Dijkstra

    朴素版Dijkstra 目标 找到从一个点到其他点的最短距离 思路 ①初始化距离dist数组,将起点dist距离设为0,其他点的距离设为无穷(就是很大的值) ②for循环遍历n次,每层循环里找出不在S ...

随机推荐

  1. nacos 集群搭建

    nacos 集群搭建 1.单机部署 从nacos官网下载zip/tar包,https://github.com/alibaba/nacos/releases/tag/2.0.2 解压后即可启动 外置数 ...

  2. Quill基本使用和配置 - DevUI

    DevUI 是一款面向企业中后台产品的开源前端解决方案,它倡导沉浸.灵活.至简的设计价值观,提倡设计者为真实的需求服务,为多数人的设计,拒绝哗众取宠.取悦眼球的设计.如果你正在开发 ToB 的工具类产 ...

  3. 『言善信』Fiddler工具 — 15、使用Fiddler抓取HTTPS请求

    目录 1.Fiddler抓取HTTPS过程 2.拓展:SSL/TLS证书握手原理 3.Fiddler抓取HTTPS原理总结 4.Fiddler抓取HTTPS设置 步骤1:配置证书 步骤2:勾选设置 5 ...

  4. docker 自定义部署Springboot——依赖与代码分离部署

    第一步:执行mvn package 命令打出jar包,然后解压jar包,把lib放到服务器合适的目录下面 第二步:打出不带jar包的SpringBoot工程 首先配置pom.xml文件 <bui ...

  5. 【dp】背包问题

    01背包 呐,为什么叫它01背包呢,因为装进去就是1,不装进去就是0.所以针对每个物品就两种状态,装,不装(请允许我用这么老套的开篇,相信听过很多次背包讲解的人,大多都是这个开篇的)所以咯,我这个背包 ...

  6. 前端 JavaScript 复制粘贴的奥义——Clipboard 对象概述

    前言 作为一名资深搬砖工,你要问我用得最熟练的技能是什么,那我敢肯定且自豪的告诉你:是 Ctrl+C !是 Ctrl+V! 不信?你来看看我键盘上的 Ctrl.C 和 V 键,那油光发亮的包浆程度,不 ...

  7. JMeter定时器种类+详细教程举例

    首先,我们先了解一下定时器的常见种类以及它的作用. 原文地址:https://www.cnblogs.com/istart/p/11184533.html 一.定时器种类+作用 上面是我截图的自己有道 ...

  8. React 开发环境准备

    1. 使用reactjs,一般有以下两种方式: (1)通过script标签引入reactjs.这种方式不推荐使用,如果我们的项目比较大,就需要对项目进行拆分,于是页面就需要通过script标签引入很多 ...

  9. 32、JavaScript介绍

    32.1.JavaScript概述: 1.JavaScript的历史: 1992年Nombas开发出C-minus-minus(C--)的嵌入式脚本语言(最初绑定在CEnvi软件中),后将其改名Scr ...

  10. CMD命令进入某个目录

    1.开始->运行->CMD 2.进入某个磁盘,直接盘符代号:如D:,不用CD 命令切换 3.进入除根录以下的文件夹 cd 文件夹路径 例如我要进入 E:/Program Files/PHP ...