算法原理

每次扩展一个距离最小的点,再更新与其相邻的点的距离。

如何寻找距离最小的点

普通的Dijkstra算法的思路是直接For i: 1 to n

优化方案是建一个小根堆,小根堆里存储由当前结点更新距离的所有点,那么堆顶就是距离最小的点

如何寻找与源点相邻的点

当然是邻接表

具体实现

建一个小根堆heap[] ,用来存储结点的序号,用一个数组pos[i] 来存储第i个结点在堆中的位置,用一个标记数组in_heap[] 来记录结点是否在堆中,dis[i] 表示到第i个结点的最短距离

对于小根堆的操作还是基本的put()get() ,但由于有的结点已经在堆中了,所以可以把put() 拆为插入堆和调整位置两个部分

完整操作如下:

1.将与源点相邻的点进行松弛操作后加入堆

2.取出位于堆顶的结点

3.若取出的点为终点,则结束算法

4.将与当前结点相邻的点进行松弛操作

​ (1)如果该点已经在堆中,就调整在堆中的位置

​ (2)如果该点不在堆中,就加入堆

5.继续第二步

例题

最短路径问题

时间限制:1秒 内存限制:256兆

题目描述

平面上有n个点(n<=100),每个点的坐标均在-10000~10000之间,其中的一些点之间有连线。若有连线,则表示可从一个点到达另一个点,即两点间有通路,通路的距离为两点间的之间距离。现在的任务是找出一点到另一点之间的最短路径、

输入

输入共有n+m+3行,其中:

第一行为整数n

第2行到第n+1行(共n行),每行两个整数x和y,描述了一个点的坐标(以一个空格分隔)

第n+2行为一个整数m,表示图中连线的个数

此后的m行,每行描述一条连线,由两个整数i和j组成,表示第i个点和第j个点之间有连线

最后一行:两个整数s和t,分别表示源点和目标点

输出

输出仅一行,一个实数(保留两位小数),表示从s到t的最短路径长度

样例输入

5

0 0

2 0

2 2

0 2

3 1

5

1 2

1 3

1 4

2 5

3 5

1 5

样例输出

3.41

代码

#include<cstdio>
#include<cstring>
#include<cmath>
#define MAXN 100+5 struct Edge//邻接表
{
int to;
double v;
Edge *next;
}; struct Node
{
int x,y;
}node[MAXN]; int n,m,pos[MAXN],heap_size,s,t,heap[MAXN];
double dis[MAXN];
Edge *first[MAXN];
bool in_heap[MAXN]; void add_edge(int u,int v,double len)
{
Edge *temp=new Edge;
temp->to=v;
temp->v=len;
temp->next=first[u];
first[u]=temp;
} void calc(int i,int j)
{
double len=sqrt(pow((double)(node[i].x-node[j].x),2)+pow((double)(node[i].y-node[j].y),2));
add_edge(i,j,len);
add_edge(j,i,len);
} void swapp(int i,int j)
{
int temp=heap[i];
heap[i]=heap[j];
heap[j]=temp;
pos[heap[j]]=j;//调整指针
pos[heap[i]]=i;
} void shift_up(int now)//调整位置
{
int next=0;
while(now>1)
{
next=now>>1;
if(dis[heap[next]]>dis[heap[now]])
swapp(next,now);
now=next;
}
} void put(int x)//插入堆
{
in_heap[x]=true;
heap[++heap_size]=x;
pos[x]=heap_size;
shift_up(heap_size);
} int get()//取堆顶元素
{
int now=1,next,res=heap[1];
in_heap[heap[1]]=false;
heap[1]=heap[heap_size--];
pos[heap[1]]=1;
while(now*2<=heap_size)
{
next=now<<1;
if(next<heap_size&&dis[heap[next+1]]<dis[heap[next]])
++next;
if(heap[now]<=heap[next])
return res;
swapp(now,next);
now=next;
}
return res;
} void dijkstra()
{
put(s);
dis[s]=0;
while(heap_size>0)
{
int top=get();
if(top==t)
break;
Edge *temp=first[top];
while(temp!=NULL)
{
if(dis[temp->to]>dis[top]+temp->v)
{
dis[temp->to]=dis[top]+temp->v;
//结点在堆中就只调整位置,否则插入堆并调整位置
if(in_heap[temp->to])
shift_up(pos[temp->to]);
else
put(temp->to);
}
temp=temp->next;
}
}
} int main()
{
int i,x,y;
scanf("%d",&n);
for(i=1;i<=n;++i)
scanf("%d%d",&node[i].x,&node[i].y);
scanf("%d",&m);
for(i=1;i<=m;++i)
{
scanf("%d%d",&x,&y);
calc(x,y);
}
scanf("%d%d",&s,&t);
memset(dis,127,sizeof(dis));
dijkstra();
printf("%.2lf\n",dis[t]);
return 0;
}

Dijkstra算法的二叉堆优化的更多相关文章

  1. 最短路径——Dijkstra算法以及二叉堆优化(含证明)

    一般最短路径算法习惯性的分为两种:单源最短路径算法和全顶点之间最短路径.前者是计算出从一个点出发,到达所有其余可到达顶点的距离.后者是计算出图中所有点之间的路径距离. 单源最短路径 Dijkstra算 ...

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

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

  3. 二叉堆(一)之 图文解析 和 C语言的实现

    概要 本章介绍二叉堆,二叉堆就是通常我们所说的数据结构中"堆"中的一种.和以往一样,本文会先对二叉堆的理论知识进行简单介绍,然后给出C语言的实现.后续再分别给出C++和Java版本 ...

  4. 二叉堆(二)之 C++的实现

    概要 上一章介绍了堆和二叉堆的基本概念,并通过C语言实现了二叉堆.本章是二叉堆的C++实现. 目录1. 二叉堆的介绍2. 二叉堆的图文解析3. 二叉堆的C++实现(完整源码)4. 二叉堆的C++测试程 ...

  5. 二叉堆(三)之 Java的实现

    概要 前面分别通过C和C++实现了二叉堆,本章给出二叉堆的Java版本.还是那句话,它们的原理一样,择其一了解即可. 目录1. 二叉堆的介绍2. 二叉堆的图文解析3. 二叉堆的Java实现(完整源码) ...

  6. 二叉堆的实现(数组)——c++

    二叉堆的介绍 二叉堆是完全二元树或者是近似完全二元树,按照数据的排列方式可以分为两种:最大堆和最小堆.最大堆:父结点的键值总是大于或等于任何一个子节点的键值:最小堆:父结点的键值总是小于或等于任何一个 ...

  7. 图论——Dijkstra+prim算法涉及到的优先队列(二叉堆)

    [0]README 0.1)为什么有这篇文章?因为 Dijkstra算法的优先队列实现 涉及到了一种新的数据结构,即优先队列(二叉堆)的操作需要更改以适应这种新的数据结构,我们暂且吧它定义为Dista ...

  8. 《Algorithms算法》笔记:优先队列(2)——二叉堆

    二叉堆 1 二叉堆的定义 堆是一个完全二叉树结构(除了最底下一层,其他层全是完全平衡的),如果每个结点都大于它的两个孩子,那么这个堆是有序的. 二叉堆是一组能够用堆有序的完全二叉树排序的元素,并在数组 ...

  9. C# 最大二叉堆算法

    C#练习二叉堆算法. namespace 算法 { /// <summary> /// 最大堆 /// </summary> /// <typeparam name=&q ...

随机推荐

  1. 远程开户系统开放API接口

    如今随着智能识别技术的成熟和商用,金融领域也开始逐渐试水"远程开户".从OCR身份证识别到人脸识别,到如今市场上即将出现完整的远程开户系统,除了需要成熟的技术做支撑外,还需要对市场 ...

  2. 2)Javascript设计模式:Singleton模式

    Singleton模式 var User = (function() { var instance; function _User(){} _User.prototype.say = function ...

  3. Spring MVC之RequestMapping

    第一部分.概述 /**映射URL到控制器类或处理程序*/@Target({ElementType.METHOD, ElementType.TYPE})@Retention(RetentionPolic ...

  4. Java解析JSON文件的方法(一)

    一.首先需要在Eclipse工程中导入相关的jar包,jar包参见链接:http://yunpan.alibaba-inc.com/share/link/NdA5b6IFK 二.提供一份待解析的jso ...

  5. js原生继承之——类式继承实例(推荐使用)

    <!DOCTYPE html><html lang="en"><head>    <meta charset="UTF-8&qu ...

  6. android 类似QQ底部输入框弹出键盘和面板冲突 布局闪动处理方案(转)

    先看下效果 差不多就是解决这种冲突,布局闪动的 作者的githup :https://github.com/Jacksgong/JKeyboardPanelSwitch Android键盘面板冲突 布 ...

  7. Xtrabackup构建MySQL主从环境

    环境:HE3主库,HE1从库 HE1:192.168.1.248 HE3:192.168.1.250 从库my.cnf加入以下参数并重启数据库: read_only=1 log_slave_updat ...

  8. C++实现具有基本功能的智能指针

    C++中的智能指针实际上是代理模式与RAII的结合. 自定义unique_ptr,主要是release()和reset().代码如下. #include <iostream> using ...

  9. svn git协同管理

    项目开发过程中总有一些奇奇怪怪的需求出现. 我们的项目管理是使用SVN的,用SVN是历史原因,无法整个项目向Git切换.由于我需要管理SVN,做一些代码合并工作.每次合并都会遇到SVN代码需要对比查看 ...

  10. mac下sublime 配置查看源码

    sublime及package control的安装查看之前一篇文章:http://www.cnblogs.com/xdwa/p/5805101.html 安装ctags 安装完毕package co ...