这个算法不能处理负边情况,有负边,请转到Floyd算法或SPFA算法(SPFA不能处理负环,但能判断负环)

SPFA(SLF优化):https://www.cnblogs.com/yifan0305/p/16391419.html

代码很长,耐下心来看完,存储方法为链式前向星存储。

(如果内存放得下的话,建议稠密图用邻接矩阵(或者跑floyd),稀疏图用邻接表,只是建议)

#include<bits/stdc++.h>
using namespace std;
int n,m,cnt;//cnt 计数器 有cnt条边(从1开始存)
struct edge
{
int u,v,w,nxt;
/*u 起点 (不需要的话可以删掉)
v 目标点 w 边权
nxt 指向这条边的上一条边的标号(上一条边是从同一起点发出的边,不要搞混了) */
};
edge e[20020];
int h[2010];//存储第x个点发出的最后一条边的编号
struct nod//存储点
{
int s,w;//s 这个点的标号 w 起点到这个点的最短路径值
};
struct cmp//结构体的比较函数(优先队列要用,看不懂去搜一下,毕竟这里不是它的主场)
{
bool operator()(nod a,nod b)
{
return a.w<b.w;
}
};
nod dis[2010];//存点(毕竟不好处理点的标号和最短路径值相匹配,如果你有更好的方法,忽略这种存法)
bool f[2010];// 标记点是否被染色(看不懂染色的往下看。。。懒得再敲一遍了)
priority_queue<nod,vector<nod>,cmp> q;//定义一个优先队列(存点)
void add(int,int,int);
void dij()//堆优化
{
for(int i=1;i<=n;++i)//初始化
{
dis[i].s=i;//将编号输入
dis[i].w=0x3f3f3f;//将值设为一个很大的值,理由是便于后续使用(废话)
}
dis[1].w=0;//起点到起点的距离是0(该题起点为1,根据题目来)
q.push(dis[1]);//该点入队
while(!q.empty())
{
nod top=q.top();//取出队首元素
int u=top.s,w=top.w;//u 该点的编号 w 起点到点u的最短路径值
f[u]=1;//标记已经被染过色了
//(染色不懂,就理解为这个点已经用过了,标记一下,不能再用了,跟搜索回溯的原理差不多)
q.pop();//出队
for(int i=h[u];i;i=e[i].nxt)//邻接表遍历
{
int v=e[i].v;//v 目标点
if(dis[v].w>w+e[i].w)//松弛操作
{
dis[v].w=w+e[i].w;//更新起点到点v的最短路径值
q.push(dis[v]);//入队
}
if(f[v]) continue;//如果已被染色(看不懂按上面的理解),跳过
//这里松弛和染色不会冲突,松弛操作是更新最优值,染色是为了防止后面扫到该点的非优值(2022.5.15修改)
}
}
}
int main()
{
scanf("%d%d",&n,&m);//输入,根据题目来
for(int a,b,c,i=1;i<=m;++i)
{
scanf("%d%d%d",&a,&b,&c);
add(a,b,c);//加边
add(b,a,c);//加边,如果是有向图,忽略这一步操作
}
dij();//调用函数
if(dis[n].w>=0x3f3f3f-1) printf("-1");
//这里是输出操作,如果起点到点n的值大于0x3f3f3f-1,就说明压根就没更新到,也说明起点和点n不连接
//(题目要求,根据题目来)
else printf("%d",dis[n].w);
return 0;
}
void add(int u,int v,int w)//加边操作
{
e[++cnt].v=v;//记录该条边的目标点
e[cnt].w=w;//记录边权
e[cnt].nxt=h[u];//记录上一条边
h[u]=cnt;//更新从点u发出的最后一条边的边号
}

重载运算符版本

#include<bits/stdc++.h>
using namespace std;
int n,m,cnt,g;//cnt 计数器 有cnt条边(从1开始存)s起点
struct edge
{
int u,v,w,nxt;
/*u 起点 (不需要的话可以删掉)
v 目标点 w 边权
nxt 指向这条边的上一条边的标号(上一条边是从同一起点发出的边,不要搞混了) */
};
edge e[20020];
int h[2010];//存储第x个点发出的最后一条边的编号
struct nod//存储点
{
int s,w;//s 这个点的标号 w 起点到这个点的最短路径值
bool operator < (const nod &a) const
{
return w>a.w;
}//运算符重载
};
nod dis[2010];//存点(毕竟不好处理点的标号和最短路径值相匹配,如果你有更好的方法,忽略这种存法)
bool f[2010];// 标记点是否被染色
priority_queue<nod> q;//定义一个优先队列(存点)
void add(int,int,int);
void dij()//堆优化
{
memset(f,0,sizeof f);//初始化数组
for(int i=1;i<=n;++i)//初始化
{
dis[i].s=i;//将编号输入
dis[i].w=0x3f3f3f;//将值设为一个很大的值
}
dis[g].w=0;//起点到起点的距离是0
q.push(dis[g]);//该点入队
while(!q.empty())
{
nod top=q.top();//取出队首元素
int u=top.s,w=top.w;//u 该点的编号 w 起点到点u的最短路径值
q.pop();//出队
if(f[u]) continue;//如果该点已经染过色了,跳过
f[u]=1;//染色
for(int i=h[u];i;i=e[i].nxt)//邻接表遍历
{
int v=e[i].v;//v 目标点
if(!f[v]&&dis[v].w>w+e[i].w)//松弛操作
{
dis[v].w=w+e[i].w;//更新起点到点v的最短路径值
q.push(dis[v]);//入队
}
}
}
}
int main()
{
scanf("%d%d%d",&n,&m,&g);//输入,根据题目来
for(int a,b,c,i=1;i<=m;++i)
{
scanf("%d%d%d",&a,&b,&c);
add(a,b,c);//加边
add(b,a,c);//加边,如果是有向图,忽略这一步操作
}
dij();//调用函数
if(dis[n].w>=0x3f3f3f-1) printf("-1");
//这里是输出操作,如果起点到点n的值大于0x3f3f3f-1,就说明压根就没更新到,也说明起点和点n不连接
else printf("%d",dis[n].w);
return 0;
}
void add(int u,int v,int w)//加边操作
{
e[++cnt].v=v;//记录该条边的目标点
e[cnt].w=w;//记录边权
e[cnt].nxt=h[u];//记录上一条边
h[u]=cnt;//更新从点u发出的最后一条边的边号
}

dijkstra最短路算法(堆优化)的更多相关文章

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

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

  2. Dijkstra最短路算法

    Dijkstra最短路算法 --转自啊哈磊[坐在马桶上看算法]算法7:Dijkstra最短路算法 上节我们介绍了神奇的只有五行的Floyd最短路算法,它可以方便的求得任意两点的最短路径,这称为“多源最 ...

  3. 【坐在马桶上看算法】算法7:Dijkstra最短路算法

           上周我们介绍了神奇的只有五行的Floyd最短路算法,它可以方便的求得任意两点的最短路径,这称为“多源最短路”.本周来来介绍指定一个点(源点)到其余各个顶点的最短路径,也叫做“单源最短路径 ...

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

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

  5. 最短路模板|堆优化Dijkstra,SPFA,floyd

    Ⅰ:Dijkstra单源点最短路 1.1Dijkstra const int MAX_N = 10000; const int MAX_M = 100000; const int inf = 0x3f ...

  6. Dijkstra算法堆优化详解

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

  7. BZOJ 3040 最短路 (堆优化dijkstra)

    这题不是裸的最短路么?但是一看数据范围就傻了.点数10^6,边数10^7.这个spfa就别想了(本来spfa就是相当不靠谱的玩意),看来是要用堆优化dijkstra了.但是,平时写dijkstra时为 ...

  8. NEU 1664 传送(最短路基础 堆优化Dijkstra)

    题目描述 小A最近喜欢上一款游戏:游戏把地图分了一些区域,这些区域可能会重叠,也可能不会. 游戏中有一项传送技能,改传送技能只能将在同一区域的两个地方使用.小A可以利用区域中重叠部分来实现从某一区域到 ...

  9. Dijkstra算法堆优化

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

随机推荐

  1. 老生常谈系列之Aop--CGLIB动态代理的底层实现原理

    老生常谈系列之Aop--CGLIB动态代理的底层实现原理 前言 上一篇老生常谈系列之Aop--JDK动态代理的底层实现原理简单讲解了JDK动态代理的实现,动态代理常用实现里面的双子星还有另一位--CG ...

  2. python使用虚拟环境venv

    venv模块支持使用自己的站点目录创建轻量级"虚拟环境",可选择与系统站点目录隔离.每个虚拟环境都有自己的Python二进制文件(与用于创建此环境的二进制文件的版本相匹配),并且可 ...

  3. Oracle 19c单实例部署

    目录 Oracle 19c单实例部署: 1.配置yum: 2.安装rpm包: 3.设置hostname: 4.配置hostname解析: 5.配置时钟同步服务(ntp): 6.检查及配置内核参数: 7 ...

  4. 【PyHacker编写指南】打造URL批量采集器

    这节课是巡安似海PyHacker编写指南的<打造URL批量采集器> 喜欢用Python写脚本的小伙伴可以跟着一起写一写呀. 编写环境:Python2.x 00x1: 需要用到的模块如下: ...

  5. 羽夏 Bash 简明教程(下)

    写在前面   该文章根据 the unix workbench 中的 Bash Programming 进行汉化处理并作出自己的整理,并参考 Bash 脚本教程 和 BashPitfalls 相关内容 ...

  6. 关于vue.config.js 的 proxy 配置

    关于vue.config.js 的 proxy 配置有的同学不怎么明白项目里面有的配置了 pathRewrite 地址重写,有的又没有进行配置?/* * proxy代理配置的说明 * *接口例子:/z ...

  7. Vue基础篇 之 v-model 模拟

    我们知道vue中 为简化表单输入 提供了v-model 的语法绑定 将 vue的属性和表单元素进行了双向绑定 大大简化了表单数据操作的数据绑定 那么v-model 是如何实现双向绑定的呢? 今天我们来 ...

  8. select into 与 insert into 的区别

    1.select * into table_A  from table_B table_A是个新创建表,table_B是个已经存在的表. 2.insert into table_A from tabl ...

  9. 机器学习-学习笔记(二) --> 模型评估与选择

    目录 一.经验误差与过拟合 二.评估方法 模型评估方法 1. 留出法(hold-out) 2. 交叉验证法(cross validation) 3. 自助法(bootstrapping) 调参(par ...

  10. UVA471 Magic Numbers 题解

    1.题目 题意很简单:输入n,枚举所有的a,b,使得 (1)满足a/b=n. (2)满足a,b各个位上的数字不相同. 2.思路 (1)对于被除数,要满足各个位上的数字,显然最大枚举到987654321 ...