dijkstra 的优先队列优化
既然要学习算法,就要学习到它的精髓,才能够使用起来得心应手。
我还是远远不够啊。
早就知道,dijkstra 算法可以用优先队列优化,我却一直不知道该怎样优化。当时,我的思路是这样的:假设有n个顶点,将这n个顶点的id和距原点的距离放在结构体内,再将这n个结构体放入优先队列中,堆顶是距源点距离最小的点。每次要更新距离时,仅仅只需要取堆顶的数就可以了。然而,具体要怎样更新堆内各点的距离呢?将堆顶取出,更新后再放回去?这样的话堆顶永远都会是同一个元素了,因为堆顶元素在更新后,还是距离最小的。那么我们可以依次取出堆顶元素,放在结构体数组之内,等待更新完毕后再放回去,那么这样的时间复杂度是2*n,而原先的时间复杂度,也是2*n,这样的优化没有意义,反而还多了一个结构体数组浪费空间。
如果你也和我的想法一样,那我们真是太有缘分了,看来大家都是蠢的七窍流血的一类人啊,你是不是也和我一样,正在反思自己是不是应该放弃学习算法啊?实际上,在dijkstra里面,有一个十分重要的标记数组,这个标记数组决定了,已经确定了最短距离的点,就不要再次优化了!你明白了吧,想想自己真是蠢呐,竟然忘记如此重要的数组!
让我们再次思考,是否要将所有没有确定的点全部放入数组呢?
当然不要,我们只要将刚刚更新过的放进去就行,因为那些没有更新的,肯定不会是路径最短的。那么我们每次都放,就会导致某个节点被放进去很多次了,但是没关系,他们的被放进去的时候,距离是不同的,所以距离大的会沉到底下去,最短路径一定不是他们(对同一节点来说),他们要出推时,我们只处理第一个,以后的一律不处理。这个我们还用一个标记数组来解决。
代码自己去找吧,https://blog.csdn.net/jobsandczj/article/details/49962557,这个人写得不错,除了码风很丑,加上竟然使用邻接矩阵。。。然后还有book数组定义了没有使用以外,其他的都还行。
如果你连这些问题都不想面对,或者根本就不想看代码的话,你还是转行吧。
在此还是贴上自己的模板吧
这个模板跑起来反而比我原先的代码更慢,我觉得这是邻接表的问题,因为在一开始我用的是啊哈算法的邻接表,而现在用的是vector,再加上可能我在做的那个题太水了,数据量太小,导致优先队列的优势没有发挥出来。
#include<iostream>
#include<algorithm>
#include<vector>
#include<queue>
using namespace std;
struct node
{
int x;
int s; bool operator<(const node p)const
{
return p.s<s;
}
};
bool book[100];
int dis[100];
const int inf = 2100000000;
int main()
{
vector<int>u[100];
vector<int>w[100];
int n,m;
cin>>n>>m;
int x,y,z;
for(int i=0;i<m;i++){
cin>>x>>y>>z;
u[x].push_back(y);
w[x].push_back(z);
}
fill(dis,dis+n+1,inf);
dis[1]=0;
priority_queue<node>q;
node exa;
exa.x=1;
exa.s=0;
q.push(exa);
while(!q.empty()){
exa=q.top();q.pop();
if(book[exa.x]){continue;}
book[exa.x]=true;
int t=exa.x;
for(int i=0;i<u[t].size();i++){
if(dis[u[t][i]]>dis[t]+w[t][i]){
dis[u[t][i]]=dis[t]+w[t][i];
exa.s=dis[u[t][i]];
exa.x=u[t][i];
q.push(exa);
}
}
}
for(int i=1;i<=n;i++){
cout<<dis[i]<<" ";
}
cout<<endl;
}
下面的代码包括了对路径的输出,只是我没有找到这样的题,所以不敢保证算法的正确性:
#include<iostream>
#include<algorithm>
#include<vector>
#include<queue>
#include<stack>
using namespace std;
struct node
{
int x;
int s; bool operator<(const node p)const
{
return p.s<s;
}
};
bool book[100];
int dis[100];
int f[1000];
const int inf = 2100000000;
int main()
{
vector<int>u[100];
vector<int>w[100];
int n,m;
cin>>n>>m;
int x,y,z;
for(int i=0;i<m;i++){
cin>>x>>y>>z;
u[x].push_back(y);
w[x].push_back(z);
}
fill(dis,dis+n+1,inf);
fill(f,f+n+1,-1);
dis[1]=0;
priority_queue<node>q;
node exa;
exa.x=1;
exa.s=0;
q.push(exa);
while(!q.empty()){
exa=q.top();q.pop();
if(book[exa.x]){continue;}
book[exa.x]=true;
int t=exa.x;
for(int i=0;i<u[t].size();i++){
if(dis[u[t][i]]>dis[t]+w[t][i]){
dis[u[t][i]]=dis[t]+w[t][i];
exa.s=dis[u[t][i]];
f[u[t][i]]=t;
exa.x=u[t][i];
q.push(exa);
}
}
}
stack<int>h;
for(int i=1;i<=n;i++){
cout<<dis[i]<<": ";
int ah=f[i]; while(ah!=-1){
h.push(ah);
ah=f[ah];
}
while(!h.empty()){
cout<<h.top()<<" ";
h.pop();
}cout<<i<<" ";
cout<<endl;
}
cout<<endl;
}
每个人都是被动出生,只有选择死亡才是真正的自由!
dijkstra 的优先队列优化的更多相关文章
- 地铁 Dijkstra(优先队列优化) 湖南省第12届省赛
传送门:地铁 思路:拆点,最短路:拆点比较复杂,所以对边进行最短路,spfa会tle,所以改用Dijkstra(优先队列优化) 模板 /******************************** ...
- poj 1511 优先队列优化dijkstra *
题意:两遍最短路 链接:点我 注意结果用long long #include<cstdio> #include<iostream> #include<algorithm& ...
- 【bzo1579】拆点+dijkstra优先队列优化+其他优化
题意: n个点,m条边,问从1走到n的最短路,其中有K次机会可以让一条路的权值变成0.1≤N≤10000;1≤M≤500000;1≤K≤20 题解: 拆点,一个点拆成K个,分别表示到了这个点时还有多少 ...
- POJ 1511 Invitation Cards(单源最短路,优先队列优化的Dijkstra)
Invitation Cards Time Limit: 8000MS Memory Limit: 262144K Total Submissions: 16178 Accepted: 526 ...
- Dijkstra算法(朴素实现、优先队列优化)
Dijkstra算法只能求取边的权重为非负的图的最短路径,而Bellman-Ford算法可以求取边的权重为负的图的最短路径(但Bellman-Ford算法在图中存在负环的情况下,最短路径是不存在的(负 ...
- dijkstra算法之优先队列优化
github地址:https://github.com/muzhailong/dijkstra-PriorityQueue 1.题目 分析与解题思路 dijkstra算法是典型的用来解决单源最短路径的 ...
- 晴天小猪历险记之Hill(Dijkstra优先队列优化)
描述 这一天,他来到了一座深山的山脚下,因为只有这座深山中的一位隐者才知道这种药草的所在.但是上山的路错综复杂,由于小小猪的病情,晴天小猪想找一条需时最少的路到达山顶,但现在它一头雾水,所以向你求助. ...
- 最短路--dijkstra+优先队列优化模板
不写普通模板了,还是需要优先队列优化的昂 #include<stdio.h> //基本需要的头文件 #include<string.h> #include<queue&g ...
- 最短路模板(Dijkstra & Dijkstra算法+堆优化 & bellman_ford & 单源最短路SPFA)
关于几个的区别和联系:http://www.cnblogs.com/zswbky/p/5432353.html d.每组的第一行是三个整数T,S和D,表示有T条路,和草儿家相邻的城市的有S个(草儿家到 ...
随机推荐
- 牛客第二场-J-farm-二维树状数组
二维树状数组真的还挺神奇的,更新也很神奇,比如我要更新一个区域内的和,我们的更新操作是这样的 add(x1,y1,z); add(x2+1,y2+1,z); add(x1,y2+1,-z); add( ...
- 状态模式-State-订单状态
JAVA设计模式-状态模式-State-订单状态 21. State(状态) 意图: 允许一个对象在其内部状态改变时改变它的行为.对象看起来似乎修改了它的类. 解释: 比如说对订单的提交,第一 ...
- 基于SSH 供应链管理系统质量属性说明
产品的易用程度如何,执行速度如何,可靠性如何,当发生异常情况时,系统如何处理.这些被称为软件质量属性,而特性是指系统非功能(也叫非行为)部分的需求. 性能:性能就是一个东西有多快,通常指响应时间或延迟 ...
- 如何实现基于ssh框架的投票系统的的质量属性
如何实现基于ssh框架的投票系统的的质量属性: 项目 :网上考试系统 我做的是网上考试系统,因为标准化的考试越来越重要,而通过计算机进行标准化判卷,系统会自动判卷出成绩,组织考试的人不用组织人员打印试 ...
- 第三个spring冲刺第9天
今天是第三阶段冲刺的最后第二天了,我们该实现的功能基本已经全部实现了,有填空的,选择题的,还有计时的,目前就是在查BUG,看看有哪些地方有BUG需要修改,以下截图是我们团队所做的功能截图: 首页: 填 ...
- JavaScript 模拟 HashMap例子
function map(){ var map = {}; // Map map = new HashMap(); var key = "key"; va ...
- 2016_NENU_CS_3
贴一下比赛的代码, 其中 I 题代码源于final大神 ok_again http://acm.hust.edu.cn/vjudge/contest/127444#overview I /***** ...
- 深入浅出——float
FLOAT 参考张鑫旭-鑫空间-鑫生活[http://www.zhangxinxu.com]的CSS float浮动的深入研究.详解及拓展 1.FLOAT的特性 float属性的初衷只是为了实现文字 ...
- python之小应用:读取csv文件并处理01数据串
目的:读取csv文件内容,把0和1的数据串取出来,统计出现1的连续次数和各次数出现的频率次数 先读取csv文件内容: import csv def csv_read(file): list = [] ...
- codeforces166E
Tetrahedron CodeForces - 166E You are given a tetrahedron. Let's mark its vertices with letters A, B ...