这个算法不能处理负边情况,有负边,请转到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. 图片查看器——viewer.js

    使用简介 https://github.com/FNNDSC/viewerjs(需要点击遮罩层关闭弹框的复制下面内容) https://www.jianshu.com/p/d98db3815823 v ...

  2. Java学习笔记-基础语法Ⅲ

    继承:子类使用extends来继承父类,子类可以有父类的内容,还可以有子类自己特有的内容 继承的好处: 提高了代码的复用性(多个类相同的成员可以放到同一个类中) 提高了代码的维护性(如果方法的代码需要 ...

  3. JavaMetaweblogClient,Metaweblog的java实现-从此上传博客实现全平台

    目录 1. 什么是Metaweblog? 2. Metaweblog的应用 3. 如何使用Metaweblog 4. 本项目介绍 4.1 metaweblog与java之间的关系映射 4.2 使用Ja ...

  4. Random 中的Seed

    C#中使用随机数 看下例 当Random的种子是0时 生成的随机数列表是一样的 也就是说当seed 一样时 审查的随机数时一样的 Random的无参实例默认 种子 时当前时间 如果要确保生成的随机数不 ...

  5. Vue基础篇之 插槽 slot

  6. 对 Python 中 GIL 的一点理解

    GIL(Global Interpreter Lock),全局解释器锁,是 CPython 为了避免在多线程环境下造成 Python 解释器内部数据的不一致而引入的一把锁,让 Python 中的多个线 ...

  7. ElasticSearch7.3学习(二十九)----聚合实战之使用Java api实现电视案例

    一.数据准备 创建索引及映射 建立价格.颜色.品牌.售卖日期字段 PUT /tvs PUT /tvs/_mapping { "properties": { "price& ...

  8. poj1475 -- Pushing Boxes

    这道题其实挺有趣 的,这让我想起小时候诺基亚手机上的推箱子游戏(虽然一点也不好玩) (英文不好-->)  题意翻译: 初始人(S),箱子(B),目的地(T)用人把箱子推到 T最小步数及其路径(满 ...

  9. 【多线程与高并发原理篇:4_深入理解synchronized】

    1. 前言 越是简单的东西,在深入了解后发现越复杂.想起了曾在初中阶段,语文老师给我们解说<论语>的道理,顺便给我们提了一句,说老子的无为思想比较消极,学生时代不要太关注.现在有了一定的生 ...

  10. 选择ERP频频踩雷?国内外ERP有差异,突破ERP软件单一性是关键

    信息化日新月异的蓬勃发展,导致企业在选择ERP软件时频频踩雷.企业如何选择出一个适合自己的ERP软件系统呢?是选择国外知名公司的ERP软件产品,还是选择国内性价比高的ERP软件产品呢,小编就带大家了解 ...