最短路径——Dijkstra算法以及二叉堆优化(含证明)
一般最短路径算法习惯性的分为两种:单源最短路径算法和全顶点之间最短路径。前者是计算出从一个点出发,到达所有其余可到达顶点的距离。后者是计算出图中所有点之间的路径距离。
单源最短路径
Dijkstra算法
思维
本质上是贪心的思想,声明一个数组dis来保存源点到各个顶点的最短距离和一个保存已经找到了最短路径的顶点的集合:S,原本的元素构成集合Q,初始时,原点 s 的路径权重被赋为 0 (dis[s] = 0)。若对于顶点 s 存在能直接到达的边(s,m),则把dis[m]设为w(s, m),同时把所有其他(s不能直接到达的)顶点的路径长度设为无穷大。初始时,集合S只有顶点s
然后,从dis数组选择最小值,则该值就是源点s到该值对应的顶点的最短路径,并且把该点加入到S中,此时完成一个顶点, 然后,我们需要看看新加入的顶点是否可以到达其他顶点并且看看通过该顶点到达其他点的路径长度是否比源点直接到达短,如果是,那么就替换这些顶点在dis中的值。 然后,又从dis中找出最小值,重复上述动作,直到S中包含了图的所有顶点。
但是很可惜,由于算法特性,一个点的距离确定后不会再改变,这里的确定不是指得到数值,而且该点被作为观察点观察后所以这个算法并不能处理带负权边的图。
但是可以利用该算法+优先队列来优化,优化后的算法打破了之前的一个点的距离确定后不会再改变的算法特性,更加类似SPFA,并且可以处理负权边。但个人觉得已经不能称作dijkstra算法了。
证明
这里给出一个命题:从集合Q中找到dis[k]最小的v,dis[k]即为源点到v的最短路径长度。
如果这个命题为真,dij的正确性就可以得证。
- 证明:从开始利用算法取得一个v1,即dis[v1]是最小的,\(\forall\)v,dis[v]>dis[v1]。
假设dis[v1]不是从源点到v1最短路径
则\(\exists\)v,使得dis[v]<dis[v1],与已知矛盾。
得证。 - 证明:已利用算法从Q中找到k个点,并确定了k个点的最短路径,此时再从Q中用算法找出一个vk+1,dis[vk+1]即为源点到vk+1最短路径。
假设dis[vk+1不是源点到vk+1的最短路径长度。
则设从源点到vk+1的最短路径经过的点的集合为V,dis为路径长度,切dis < dis[vk+1]。
设V中最靠近vk+1且不属于S的点为vx,vx的后继点为vy
。如果有向图中皆为正权边,则易得dis[vx] < dis[vy] <= dis。(vy = vk+1时等号成立)
但又因为vx不属于S,则dis[vx] > dis,矛盾。
得证。 - 综上所述,命题得证。
- 负权边的时候,dis[vx] < dis[vy]和dis[vx] > dis都不一定成立。故不得证。
举例演算

| 集合S | 当前观察点u | dis[2] | dis[3] | dis[4] | dis[5] | dis[6] |
|---|---|---|---|---|---|---|
| 1 | - | 1 | ∞ | 2 | ∞ | ∞ |
| 1,2 | 2 | 1 | ∞ | 2 | 4 | 6 |
| 1,2,4 | 4 | 1 | 5 | 2 | 4 | 6 |
| 1,2,4,5 | 5 | 1 | 5 | 2 | 4 | 6 |
| 1,2,4,5,3 | 3 | 1 | 5 | 2 | 4 | 5 |
| 1,2,4,5,3,6 | 6 | 1 | 5 | 2 | 4 | 5 |
从结点1出发,1与2、4连通,确定(1,2),(1,4)的距离,其中到2的距离最短,再从观察点2出发,2与5、6连通,根据dis[u]+c[u][v]<dis[v]的判断关系出发,更新dis。接着再分别从观察点4,5,3,6出发更新dis,得到最终的结点1的单源最短路径。
代码实现
朴素dij算法,时间复杂度约为O(n2)
#include<iostream>
#include<algorithm>
#include<cmath>
#include<vector>
#include<queue>
using namespace std;
const int maxn = 1000;
const int inf = 0x7fffffff;
int n, m;
int e[maxn][maxn], dis[maxn];
int book[maxn];
void dij(int s) {
for (int i = 1; i <= n; i++) dis[i] = e[s][i];
for (int i = 1; i <= n; i++) book[i] = 0;
book[s] = 1;
dis[s] = 0;
for (int i = 1; i <= n - 1; i++) {
int min = inf;
int u;
for (int j = 1; j <= n; j++) {
if (book[j] == 0 && dis[j] < min) {
min = dis[j];
u = j;
}
}
book[u] = 1;
for (int v = 1; v <= n; v++) {
if (e[u][v] < inf && book[v] == 0) {
if (dis[v] > dis[u] + e[u][v]) {
dis[v] = dis[u] + e[u][v];
}
}
}
}
}
int main() {
cin >> n >> m;
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= n; j++) {
e[i][j] = inf;
}
}
for (int i = 0; i < m; i++) {
int u, v, w;
cin >> u >> v >> w;
e[u][v] = w;
}
int x;
cin >> x;
dij(x);
for (int i = 1; i <= n; i++) {
cout << dis[i] << " ";
}
return 0;
}
为了能够方便的寻找当前最小的dis作为观察点,可以利用优先队列最小堆来优化。同时利用初始化建表来节省寻找每个点的相邻点的过程。这里使用的是stl中的优先队列,底层是heap实现的,应该是二叉堆,所以时间复杂度应该是O((m+n)logn),如果是使用斐波那契堆,可以到O(nlogn)。
#include<iostream>
#include<algorithm>
#include<cmath>
#include<vector>
#include<queue>
using namespace std;
const int MAX = 1000;
int h[MAX * 2], to[MAX * 2], nxt[MAX * 2], co[MAX * 2], dis[MAX], k = 0, n, m;
priority_queue<pair<int, int>, vector<pair<int, int> >, greater<pair<int, int> > > que;
void insert(int u,int v,int c) {
nxt[++k] = h[u];
h[u] = k;
to[k] = v;
co[k] = c;
nxt[++k] = h[v];
h[v] = k;
to[k] = u;
co[k] = c;
}
void dij(int s) {
for (int i = 0; i < MAX; i++) dis[i] = 0x7FFFFFFF;
dis[s] = 0;
que.push(make_pair(dis[s], s));
while (!que.empty()) {
pair<int, int> u = que.top();
que.pop();
if (dis[u.second] < u.first) continue;
for (int i = h[u.second]; i; i = nxt[i]) {
if (dis[to[i]] > dis[u.second] + co[i]) {
dis[to[i]] = dis[u.second] + co[i];
que.push(make_pair(dis[to[i]], to[i]));
}
}
}
}
int main() {
cin >> n >> m;
memset(h, 0, sizeof(h));
int u, v, c;
for (int i = 0; i < m; i++) {
cin >> u >> v >> c;
insert(u, v, c);
}
int x;
cin >> x;
dij(x);
for (int i = 1; i <= n; i++) {
if (dis[i] > 100000) cout << "none" << " ";
else cout << dis[i] << " ";
}
cout << endl;
return 0;
}
最短路径——Dijkstra算法以及二叉堆优化(含证明)的更多相关文章
- Dijkstra算法的二叉堆优化
Dijkstra算法的二叉堆优化 算法原理 每次扩展一个距离最小的点,再更新与其相邻的点的距离. 如何寻找距离最小的点 普通的Dijkstra算法的思路是直接For i: 1 to n 优化方案是建一 ...
- POJ 3635 - Full Tank? - [最短路变形][手写二叉堆优化Dijkstra][配对堆优化Dijkstra]
题目链接:http://poj.org/problem?id=3635 题意题解等均参考:POJ 3635 - Full Tank? - [最短路变形][优先队列优化Dijkstra]. 一些口胡: ...
- 二叉堆(一)之 图文解析 和 C语言的实现
概要 本章介绍二叉堆,二叉堆就是通常我们所说的数据结构中"堆"中的一种.和以往一样,本文会先对二叉堆的理论知识进行简单介绍,然后给出C语言的实现.后续再分别给出C++和Java版本 ...
- 二叉堆(二)之 C++的实现
概要 上一章介绍了堆和二叉堆的基本概念,并通过C语言实现了二叉堆.本章是二叉堆的C++实现. 目录1. 二叉堆的介绍2. 二叉堆的图文解析3. 二叉堆的C++实现(完整源码)4. 二叉堆的C++测试程 ...
- 二叉堆(三)之 Java的实现
概要 前面分别通过C和C++实现了二叉堆,本章给出二叉堆的Java版本.还是那句话,它们的原理一样,择其一了解即可. 目录1. 二叉堆的介绍2. 二叉堆的图文解析3. 二叉堆的Java实现(完整源码) ...
- 二叉堆的实现(数组)——c++
二叉堆的介绍 二叉堆是完全二元树或者是近似完全二元树,按照数据的排列方式可以分为两种:最大堆和最小堆.最大堆:父结点的键值总是大于或等于任何一个子节点的键值:最小堆:父结点的键值总是小于或等于任何一个 ...
- 图论——Dijkstra+prim算法涉及到的优先队列(二叉堆)
[0]README 0.1)为什么有这篇文章?因为 Dijkstra算法的优先队列实现 涉及到了一种新的数据结构,即优先队列(二叉堆)的操作需要更改以适应这种新的数据结构,我们暂且吧它定义为Dista ...
- 《Algorithms算法》笔记:优先队列(2)——二叉堆
二叉堆 1 二叉堆的定义 堆是一个完全二叉树结构(除了最底下一层,其他层全是完全平衡的),如果每个结点都大于它的两个孩子,那么这个堆是有序的. 二叉堆是一组能够用堆有序的完全二叉树排序的元素,并在数组 ...
- C# 最大二叉堆算法
C#练习二叉堆算法. namespace 算法 { /// <summary> /// 最大堆 /// </summary> /// <typeparam name=&q ...
随机推荐
- java.util.ArrayList,java.util.LinkedList,java.util.Vector的区别,使用场合.
下图是Collection的类继承图 从图中可以看出:Vector.ArrayList.LinkedList这三者都实现了List 接口.所有使用方式也很相似,主要区别在于实现方式的不同,所以对不同的 ...
- MySql学习笔记06
课程回顾 一对一关联 案例1:查询每个员工的名字和主管领导的名字 select e.ename 员工姓名,m.ename 领导姓名 from emp e join emp m on e.mgr=m.e ...
- DevOps - 版本控制 - Git
配置 .gitignore 配置 .gitignore 配置文件用于配置不需要加入版本管理的文件,配置好该文件可以为我们的版本管理带来很大的便利. 有些时候,你必须把某些文件放到Git工作目录中,但又 ...
- 汇编:将指定的内存中连续N个字节填写成指定的内容
1.loop指令实现 ;=============================== ;循环程序设计 ;将制定内存中连续count个字节填写成指定内容(te) ;loop指令实现 DATAS SEG ...
- sax技术解析xml下jaxp解析器详细代码
*解析xml的两种技术dom和sax dom:根据xml的层级结构在内存中分配一个树形结构,把xml标签,属性,文本封装成对象. sax方式:事件驱动,边读边解析. 在javax.xml.parser ...
- Xcode升到7.1插件失效解决方法
Mac前段时间下载了新的OS系统与Xcode 7.1,然而在使用Xcode 7.1时,发现插件不能用了,瞬间木有爱了,正好交流群里有人问到了插件失效的问题,经过各路大神的神通最终用下面这种方法完美解决 ...
- node 写api几个简单的问题
最近出了一直在做无聊的管理后台,还抽空做了我公司的计费终端,前端vue,后端node,代码层面没啥太多的东西.由于自己node版本是8.0.0,node自身是不支持import和export的,要想基 ...
- 终于搞定了cxgrid的多行表头(转终于搞定了cxgrid的多行表头 )
终于搞定了cxgrid的多行表头 转自:http://mycreature.blog.163.com/blog/static/556317200772524226400/ 这一周都在处理dbg ...
- labview在编写程序框图中遇到的一些布尔按钮控制布尔指示灯问题
上图布尔控件按下,数据0x04成功发送给下位机,布尔灯不亮. ............... ............. ........... 下图布尔控件按下, ...
- 让CPU使用率正弦变化
网络上流传一个面试题,说如何编程让CPU的使用率按照正弦方式变化 代码如下(运行环境Linux): #include <stdio.h> #include <stdlib.h> ...