Dijkstra算法

Dijkstra算法

大家好,又见面了,我是全栈君,祝每个程序员都可以多学几门语言。

Dijkstra(迪杰斯特拉)算法是典型的最短路径路由算法,用于计算一个节点到其它全部节点的最短路径。主要特点是以起始点为中心向外层层扩展,直到扩展到终点为止。Dijkstra算法能得出最短路径的最优解,但因为它遍历计算的节点非常多,所以效率低。

  Dijkstra算法是非常有代表性的最短路算法,在非常多专业课程中都作为基本内容有具体的介绍,如数据结构,图论,运筹学等等。

其基本思想是,设置顶点集合S并不断地作贪心选择来扩充这个集合。一个顶点属于集合S当且仅当从源到该顶点的最短路径长度已知。

初始时,S中仅含有源。设u是G的某一个顶点,把从源到u且中间仅仅经过S中顶点的路称为从源到u的特殊路径,并用数组dist记录当前每一个顶点所相应的最短特殊路径长度。Dijkstra算法每次从V-S中取出具有最短特殊路长度的顶点u,将u加入�到S中,同一时候对数组dist作必要的改动。一旦S包括了全部V中顶点,dist就记录了从源到全部其他顶点之间的最短路径长度。

比如,对下图中的有向图,应用Dijkstra算法计算从源顶点1到其他顶点间最短路径的过程列在下表中。

Dijkstra算法
Dijkstra算法的迭代过程:

Dijkstra算法

eg:

在每年的校赛里,全部进入决赛的同学都会获得一件非常美丽的t-shirt。可是每当我们的工作人员把上百件的衣服从商店运回到赛场的时候,却是非常累的!所以如今他们想要寻找最短的从商店到赛场的路线,你能够帮助他们吗?
 
Input
输入包含多组数据。每组数据第一行是两个整数N、M(N<=100,M<=10000),N表示成都的大街上有几个路口,标号为1的路口是商店所在地,标号为N的路口是赛场所在地,M则表示在成都有几条路。N=M=0表示输入结束。接下来M行,每行包含3个整数A,B,C(1<=A,B<=N,1<=C<=1000),表示在路口A与路口B之间有一条路,我们的工作人员须要C分钟的时间走过这条路。 输入保证至少存在1条商店到赛场的路线。
 
Output
对于每组输入,输出一行,表示工作人员从商店走到赛场的最短时间
 
Sample Input
2 1

1 2 3

3 3

1 2 5

2 3 5

3 1 2

0 0

 
Sample Output
3 2

#include<iostream>
#include<stdio.h>
#include<iomanip>
using namespace std;
#define N 10000
#define MAX 100000099
int a[N][N];
int dist[N];

void input(int n,int m)
{
	int p,q,len,i,j;
	for(i=1;i<=n;i++)
	{
		for(j=1;j<=n;j++)
			a[i][j]=MAX;
		dist[i]=MAX;
	}
	for(i=0;i<m;i++)
	{
		cin>>p>>q>>len;
		if(len<a[p][q])
		{
			a[p][q]=len;
			a[q][p]=len;
		}
	}
}

void dijkstra(int n)
{
	int s[N],newdist,i;
	for(i=1;i<=n;i++)
	{
		dist[i]=a[1][i];
		s[i]=0;
	}
	dist[1]=0;
	s[1]=1;
	for(i=2;i<=n;i++)
	{
		int j,tem=MAX;
		int u=1;
		for(j=2;j<=n;j++)
			if(!s[j]&&dist[j]<tem)
			{
				u=j;
				tem=dist[j];
			}
		s[u]=1;
		for(j=2;j<=n;j++)
		{
			if(!s[j]&&a[u][j]<MAX)
			{
				newdist=dist[u]+a[u][j];
				if(newdist<dist[j])
					dist[j]=newdist;
			}
		}
	}
}

int main()
{
	int n,m;
	while(scanf("%d%d",&n,&m),m||n)
	{
		input(n,m);
		dijkstra(n);
		cout<<dist[n]<<endl;
	}
	return 0;
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/118602.html原文链接:https://javaforall.net

(0)
上一篇 2021年12月3日 下午10:00
下一篇 2021年12月3日 下午11:00


相关推荐

  • SMO算法最通俗易懂的解释

    SMO算法最通俗易懂的解释我的机器学习教程「美团」算法工程师带你入门机器学习已经开始更新了,欢迎大家订阅~任何关于算法、编程、AI行业知识或博客内容的问题,可以随时扫码关注公众号「图灵的猫」,加入”学习小组“,沙雕博主在线答疑~此外,公众号内还有更多AI、算法、编程和大数据知识分享,以及免费的SSR节点和学习资料。其他平台(知乎/B站)也是同名「图灵的猫」,不要迷路哦~SVM通常用对偶问题来求解,这…

    2022年6月30日
    33
  • matlab求维纳的年龄,维纳过程及matlab图像应用

    matlab求维纳的年龄,维纳过程及matlab图像应用维纳随机过程一 概念和理论知识分析维纳过程是一类非常重要的随机过程 它是基于对粒子布朗运动的数学刻画 维纳过程经常被广泛地应用到经济学 管理学等其他应用学科之中 1 其定义为 2 若独立增量过程 W t 其增量的概率分布服从高斯分布 w2 w1 2fw w2 w1 t1 t2 2 t2 t1 0 1 正态过程 W t 的起始值和均值皆为 0 W

    2026年3月19日
    3
  • 利用 Composer 一步一步构建自己的 PHP 框架(四)——使用 ORM

    利用 Composer 一步一步构建自己的 PHP 框架(四)——使用 ORM

    2021年11月7日
    38
  • 检测Chrome headless的技巧

    检测Chrome headless的技巧原文链接 https antoinevaste com bot 20detection 2018 01 17 detect chrome headless v2 html 更新 我创建了一个库 可以通过浏览器指纹来检测脚本和爬虫 这个库仍然在开发过程中 不过你可以开始尝试使用了 代码已经在 Github 上了 目前 我在测试一个新的检测方法 欢迎来进行挑战 并给出反馈 之前我写了一个博

    2026年3月17日
    2
  • docker 镜像构建_如何更新docker镜像内的文件

    docker 镜像构建_如何更新docker镜像内的文件前言如果我们已经安装了一个python3的环境,如果另一台机器也需要安装同样的环境又要敲一遍,很麻烦,这里可以配置Dockerfile文件,让其自动安装,类似shell脚本Dockerfile编写

    2022年7月31日
    14
  • 用 DeepSeek 打造你的超强代码助手

    用 DeepSeek 打造你的超强代码助手

    2026年3月15日
    4

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号