这个算法不能处理负边情况,有负边,请转到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. Markdown基础语法(上)

    前言 按照官方文档,和根据自己所用和所理解所写 一.标题语法 一级标题最大,六级标题最小 # 一级标题 ## 二级标题 ### 三级标题 #### 四级标题 ##### 五级标题 ###### 六级标 ...

  2. Mqtt入门:在线调试连接阿里云

    近期课设需要做个东西,我想要做个上位机更好的显示. 但是一开始学习一样东西,听没头绪的,不知道从哪里入手,尝试过去B站找视频看,但是感觉视频讲的都是基础的东西,不是说人家up主讲的不好,只是对于入门, ...

  3. 镜头随人物而动,视频编辑服务让用户稳站C位

    现如今,视频是用户记录生活最热门的方式,各种App在发布视频界面都提供了视频简单剪辑的功能.除了增加音乐.滤镜.贴纸这些基础功能以外,用户越来越追求镜头感,这往往需要通过专业的视频剪辑软件手动打上关键 ...

  4. linux篇-Centos7构建NFS服务器和连接

    准备两台centos7虚拟机 192.168.30.133 192.168.30.129 2.192.168.30.1(服务端), 3查看rpc服务是否启动 4测试安装是否成功 5修改配置文件vi/e ...

  5. Fail2ban 配置详解 过滤器配置

    Fail2ban自带了很多相关服务日志的过滤器. ### # 包含配置 ### [INCLUDES] before = common.conf # 还包含其他文件中的配置,在加载本配置文件中配置之前先 ...

  6. 【单片机】CH32V103C8T6定时器3程序实验

    代码功能:每隔1毫秒进入一次定时器中断. 每隔1秒串口打印一次数据. time.c #include "time.h" #include "ch32v10x.h" ...

  7. 『忘了再学』Shell基础 — 17、预定义变量

    目录 1.预定义变量$? 2.预定义变量$$和$! 我们之前说过,Shell中的变量不是按照变量值的类型来进行分类的,而是按照Linux系统中定义的变量类别来分类的. 预定义变量就是,事先把变量的名称 ...

  8. hexo + typora 图片插入解决办法

    Typora 是一款知名的 Markdown 编辑器,简单好用,体验良好.使用 hexo 搭建好博客后,主要是用 Markdown 来编写博客,typora 便是我的首选编辑器.但直接使用 typor ...

  9. 『忘了再学』Shell基础 — 24、Shell正则表达式的使用

    目录 1.正则表达式说明 2.基础正则表达式 3.练习 (1)准备工作 (2)*练习 (3).练习 (4)^和$练习 (5)[]练习 (6)[^]练习 (7)\{n\}练习 (8)\{n,\}练习 ( ...

  10. c++ FHQ Treap

    前面我们对平衡树有了个大概的了解 关于 Treap Treap=Binary Search Tree + Heap 二叉搜索树 + 二叉堆(一般是小根堆) Treap 每一个节点有两个值 一个值是平衡 ...