【数据结构】图论中求最短路径——迪杰斯特拉算法(Dijkstra)、弗洛伊德算法(Floyd)

news2024/10/6 8:26:29

目录

  • 最短路径 (*)
    • 迪杰斯特拉算法(Dijkstra)
      • 迪杰斯特拉算法(Dijkstra)的算法原理:
    • 弗洛伊德算法(Floyd)
      • 弗洛伊德算法(Floyd)的算法原理:
      • 弗洛伊德算法的(c语言)完整实例:

最短路径 (*)

生活中最短路径问题例如:
交通网络:给定了该网内的n个城市以及这些市之间的相通公路的距离,能否找到城市A城市B之间一条最近的通路呢?

  1. 从A地到B地换车次数最少的路径
  2. 从A地到B地最短的路径(距离最短,行驶时间最短,费用最低)
  1. 迪杰斯特拉(Dijkstra)算法–从一个源点到其它各点的最短路径
  2. 弗洛伊德(Floyd)算法–每一对顶点之间的最短路径
  3. Bellman-Ford算法

迪杰斯特拉算法(Dijkstra)

该算法只适用于静态网络网络上边的权值不能为负数

基本思想:设集合S中存放已找到最短路径的顶点,集合 T = V − S T =V-S TVS存放当前还未找到最短路径的顶点。
1.初态: S中 只包含源点 v0v0到其余 各点的弧 为各点当前各点的“最短”路径。
2.从T中选取当前各点的“最短”路径长度中最短的顶点u加入到S中。
3.S加入新的顶点u后,考察顶点 v 0 v_0 v0T中剩余顶点的最短路径长度是否可以优化更新:T中各顶点新的最短路径长度值为原来的最短路径长度值、顶点u的最短路径长度值加上u到该顶点的路径长度值中的较小值。
4.重复2,3,直到T的顶点全部加入到S中、或源点到剩余顶点的路径都是为止。

在这里插入图片描述

一、图的存储:邻接矩阵和邻接表都可以

#define max 100
typedef struct {
	int arcs[max][max];
	int vexnum,arcnum;
}AGraphs;
Agraphs G;//定义图存储结构  邻接矩阵的存储形式

二、区分已经求出最短路径的点

方法一:设一个一维数组int final[max];
final[i]=1表示从源点到顶点i的最短路径已经求出,iS
final[i]=0表示从源点到顶点i的最短路径尚未求出,iV-S

方法二:利用邻接矩阵主对角线的位置G.arcs[i][i]表示i是否在S
G.arcs[i][i]=1表示从源点到顶点i的最短路径已经求出iS
G.arcs[i][i]=0表示从源点到顶点i的最短路径尚未求出iV-S

三、表示源点到顶点i最短路径

一维数组int D[max]表示最短路径的长度
D[i] :从源点到点 v i v_i vi的最短路径的长度
初态为:若从源点 到 v i v_i vi有弧,则D[i]为弧上的权值;否则置 D[i] ,即:D[i]=G.arcs[k][i]; //说明:k为源点

二维数组int P[max][max]表示最短路径包含的顶点

P[i][ ] :从源点到点 v i v_i vi的最短路径

P[i][j]=0 v j v_j vj不在从源点 到点 v i v_i vi的最短路径上

P[i][j]=1 v j v_j vj位于从源点 到点 v i v_i vi的最短路径上。

迪杰斯特拉算法(Dijkstra)的算法原理:


void ShortestPath(AGraphs G,int k,int P[][], int D[]){ 
	int i,w, j,min;
	
	for (i=0;i<G.vexnum; i ++){  
	
		final[i]=0; //初始化
		
		D[i]=G.arcs[k][i];//最短路径长度
		
		for(w=0;w<G.vexnum; w ++) 
			P[i][w]=0;//初始化
			
		if (D[i]<INFINITY){ //短路径包含的顶点
			P[i][k]=1; 
			P[i][i]=1; 
		}
			
	}
	
	D[k]=0; //初始点
	
	final[k]=1;
	
	for(i=1; i<G.vexnum; i ++){  
	
		min=INFINITY;//初始化为 无穷大
		
		for (w=0;w<G.vexnum; w ++)
			if (!final[w]&&D[w]<min){//
				j=w; 
				min=D[w];
			} 
			
		if(min== INFINITY) //
			return;
			
		final[j]=1; //标记为选入
		
		for(w=0;w<G.vexnum; w ++)
			if(!final[w]&&(min+G.arcs[j][w]<D[w])){ 
				D[w]=min+G.arcs[j][w];
				P[w]=P[j]; //??
				P[w][w]=1;  
			}
			
	}

弗洛伊德算法(Floyd)

在这里插入图片描述

图的存储:邻接矩阵和邻接表都可以

#define max 100
typedef struct {
	int arcs[max][max];
	int vexnum,arcnum;
}AGraphs;
Agraphs G;//定义图存储结构  邻接矩阵的存储形式

弗洛伊德算法(Floyd)的算法原理:

void s1(int D[][],int p[][][], Agraphs G){   

	int i,j,k;
	
	for(i=0;i<G. vexnum;i++) 
		for(j=0;j<G. vexnum;j++){  
			D[i][j]=G.arcs[i][j];
			for(k=0;k<G.vnum;k++) 
				p[i][j][k]=0; //初始化
				if(D[i][j]<INFINITY){ 
					p[i][j][i]=1; //初始化
					p[i][j][j]=1; //初始化
				}
		}
		
	for (k=0;k<G. vexnum;k++) 
		for (i=0;i<G. vexnum;i++)
			for(j=0;j<G vexnum;j++) 
				if(D[i][k]+D[k][j]<D[i][j]){//是否K 加入能够减小路径长度
					D[i][j]=D[i][k]+D[k][j];//如果可以 就加入
					for(w=0;w<G. vexnum;w++) 
						p[i][j][w]=p[i][k][w]||p[k][j][w];//然后确定路径上的元素
				}
}

弗洛伊德算法的(c语言)完整实例:

//算法6.11 弗洛伊德算法

#include  <iostream>
using  namespace  std;

#define  MaxInt  32767       //表示极大值,即∞
#define  MVNum  100           //最大顶点数

typedef  char  VerTexType;  //假设顶点的数据类型为字符型  
typedef  int  ArcType;   //假设边的权值类型为整型  

int  Path[MVNum][MVNum];                                                //最短路径上顶点vj的前一顶点的序号
int  D[MVNum][MVNum];                                                //记录顶点vi和vj之间的最短路径长度

//------------图的邻接矩阵---------------
typedef  struct{  
        VerTexType  vexs[MVNum];    //顶点表  ?
        ArcType  arcs[MVNum][MVNum];//邻接矩阵  
        int  vexnum,arcnum;         //图的当前点数和边数  
}AMGraph;



int  LocateVex(AMGraph  G  ,  VerTexType  v){
        //确定点v在G中的位置
        for(int  i  =  0;  i  <  G.vexnum;  ++i)
                if(G.vexs[i]  ==  v)
                        return  i;
                return  -1;
}//LocateVex

void  CreateUDN(AMGraph  &G){  
        //采用邻接矩阵表示法,创建有向网G  
        int  i , j  , k;
        //cout  <<"请输入总顶点数,总边数,以空格隔开:";
        cin  >>  G.vexnum  >>  G.arcnum;                                                        //输入总顶点数,总边数

        //cout  <<  "输入点的名称,如a"  <<  endl;

        for(i  =  0;  i  <  G.vexnum;  ++i){      
                //cout  <<  "请输入第"  <<  (i+1)  <<  "个点的名称:";
                cin  >>  G.vexs[i];                                                                        //依次输入点的信息  
        }
        for(i  =  0;  i  <  G.vexnum;  ++i){                                                        //初始化邻接矩阵,边的权值均置为极大值MaxInt  
                for(j  =  0;  j  <  G.vexnum;  ++j){    
                        if(j  !=  i)
                            G.arcs[i][j]  =  MaxInt;    
                        else
                            G.arcs[i][j]  =  0;
                }//for
        }//for  //初始化

        //cout  <<  "输入边依附的顶点及权值,如a  b  3"  <<  end;

        for(k  =  0;  k  <  G.arcnum;++k){                                                //构造邻接矩阵  
            VerTexType  v1  ,  v2;
            ArcType  w;

        //cout  <<  "请输入第"  <<  (k  +  1)  <<  "条边依附的顶点及权值:";
            cin  >>  v1  >>  v2  >>  w;                                                      //输入一条边依附的顶点及权值
            i  =  LocateVex(G,  v1);    j  =  LocateVex(G,  v2);        //确定v1和v2在G中的位置,即顶点数组的下标  
            G.arcs[i][j]  =  w;                                                                //边<v1,  v2>的权值置为w  
        }//for
}//CreateUDN  

/*
【样例输入】

4 5

A B C D 

A B 2 

A C 4

B C 3

B D 5

C D 1

A D

【样例输出】

5

*/

void  ShortestPath_Floyed(AMGraph  G){   

int i, j, k;

	for( i=0;i<G.vexnum;i++ )
	{
		for( j=0;j<G.vexnum;j++ )
		{
			D[i][j] = G.arcs[i][j];	// 距离矩阵初始化
			
                        Path[i][i] = i;		// 路径矩阵初始化
                        
                        if(G.arcs[i][j]!=MaxInt){
                                Path[i][j]=i;
                        }
		}
	}


	for( k=0;k<G.vexnum;k++ )     
	{
		for( i=0;i<G.vexnum;i++ )
		{
			for( j=0;j<G.vexnum;j++ )
			{
				if( D[i][k] + D[k][j] < D[i][j] )
				{
					D[i][j] = D[i][k] + D[k][j];   // 动态更新距离矩阵
					Path[i][j] = Path[k][j];       // 动态更新路径矩阵
				}
			}
		}
	}

        //test   看每个矩阵的结果

        // 	printf("\nG.arc矩阵的结果如下:\n");
	// for( i=0;i<G.vexnum;i++ )     // 输出
	// {
	// 	for( j=0;j<G.vexnum;j++ )
	// 	{
	// 		printf("%d  ",G.arcs[i][j]);
	// 	}
	// 	printf("\n");
	// }

        // 	printf("\n距离矩阵的结果如下:\n");
	// for( i=0;i<G.vexnum;i++ )     // 输出
	// {
	// 	for( j=0;j<G.vexnum;j++ )
	// 	{
	// 		printf("%d  ",D[i][j]);
	// 	}
	// 	printf("\n");
	// }

	// printf("\n路径矩阵的结果如下:\n");
	// for( i=0;i<G.vexnum;i++ )     // 输出
	// {
	// 	for( j=0;j<G.vexnum;j++ )
	// 	{
	// 		printf("%d  ",Path[i][j]);
	// 	}
	// 	printf("\n");
	// }

}


void  DisplayPath(AMGraph  G  ,  int  begin  ,int  temp  ){
        //显示最短路径
        if(Path[begin][temp]  !=  -1){
                DisplayPath(G  ,  begin  ,Path[begin][temp]);
                cout  <<  G.vexs[Path[begin][temp]]  <<  "-->";
        }
}//DisplayPath

int  main(){
        //cout  <<  "************算法6.11 弗洛伊德算法**************"  <<  endl  <<  endl;
        AMGraph  G;
        char  start  ,  destination;
        int  num_start  ,  num_destination;

        CreateUDN(G);
        
        //test
        //cout  <<  "有向网G创建完成!"  <<  endl;
        //test

        //需要完成的函数
        ShortestPath_Floyed(G);
        //需要完成的函数

        //test
        //cout  <<  "请依次输入路径的起点与终点的名称:";
        //test

        cin  >>  start  >>  destination;
        num_start  =  LocateVex(G  ,  start);
        num_destination  =  LocateVex(G  ,  destination);

        //DisplayPath(G  ,  num_start  ,  num_destination);
        //cout  <<  G.vexs[num_destination]  <<  endl;

        cout  <<  D[num_start][num_destination]  <<  endl;

        return  0;
}//main 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1788986.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Ubuntu server 24 (Linux) 普通用户不能sudo 也不能使用root登录 忘记root密码 修复解决方案

一 普通用户无法sudo&#xff0c;同时也没有其他用户可用 #test用户使用sudo报错&#xff0c;没有权限 testtest:~$ sudo vi /etc/sudoers [sudo] password for test: test is not in the sudoers file. 二 关闭ubuntu 服务器&#xff0c;重新开机 按下ESC 键 1 出现GRUB…

LLM推理加速原理(一)

1.大语言模型的基本结构 transfomer block: 输入--->正则化-->qkv三个矩阵层(映射到三个不同空间中)---->q,k,v之后self attention进行三0合一---->线性映射,正则化。 2.大语言模型的推理 目前主流的语言大模型都采用decoder-only的结构,其推理过程由两部分…

线性代数|机器学习-P2 A的列向量空间

文章目录 1. Ax矩阵的形式2. ACR 矩阵分解2.1 rank1 矩阵分解2.2 rank2 矩阵分解2.3 ACMR,求M 3. Ax 向量 1. Ax矩阵的形式 假设我们有如下矩阵A&#xff1a; A x [ 2 1 3 3 1 4 5 7 12 ] [ x 1 x 2 x 3 ] \begin{equation} Ax\begin{bmatrix} 2&1&3\\\\ 3&1&am…

【Redis数据库百万字详解】数据持久化

文章目录 一、持久化1.1、什么是持久化1.2、持久化方式1.3、RDB优缺点1.4、AOF优缺点 二、RDB持久化触发机制2.1、手动触发2.2、自动触发 三、RDB持久化配置3.1、配置文件3.2、配置查询/设置3.3、禁用持久化3.4、RDB文件恢复 四、RDB持久化案例4.1、手动持久化4.2、自动持久化案…

【计算机网络】对应用层HTTP协议的重点知识的总结

˃͈꒵˂͈꒱ write in front ꒰˃͈꒵˂͈꒱ ʕ̯•͡˔•̯᷅ʔ大家好&#xff0c;我是xiaoxie.希望你看完之后,有不足之处请多多谅解&#xff0c;让我们一起共同进步૮₍❀ᴗ͈ . ᴗ͈ აxiaoxieʕ̯•͡˔•̯᷅ʔ—CSDN博客 本文由xiaoxieʕ̯•͡˔•̯᷅ʔ 原创 CSDN 如…

超越传统AI 新型多智能体系统MESA,探索效率大幅提升

探索多智能体强化学习的协同元探索 —— MESA 算法深度解读在多智能体强化学习&#xff08;MARL&#xff09;的征途中&#xff0c;如何高效探索以发现最优策略一直是研究者们面临的挑战。特别是在稀疏奖励的环境中&#xff0c;这一问题变得更加棘手。《MESA: Cooperative Meta-…

【Vue】v-model在其他表单元素的使用

讲解内容&#xff1a; 常见的表单元素都可以用 v-model 绑定关联 → 快速 获取 或 设置 表单元素的值 它会根据 控件类型 自动选取 正确的方法 来更新元素 输入框 input:text ——> value 文本域 textarea ——> value 复选框 input:checkbox ——> checked…

Java 垃圾回收

文章目录 1 Java 垃圾回收1.1 JVM1.2 Java 对象生命周期 2 如何判断一个对象可被回收2.1 引用计数算法2.2 可达性分析算法 3 垃圾回收过程3.1 总体过程3.2 为什么要进行世代垃圾回收&#xff1f;3.3 分代垃圾回收过程 在 C 和 C 中&#xff0c;许多对象要求程序员声明他们后为其…

SpringMVC:拦截器(Interceptor)

1. 简介 拦截器&#xff08;Interceptor&#xff09;类似于过滤器&#xff08;Filter&#xff09; Spring MVC的拦截器作用是在请求到达控制器之前或之后进行拦截&#xff0c;可以对请求和响应进行一些特定的处理。拦截器可以用于很多场景下&#xff1a; 1. 登录验证&#xf…

Facebook开户|如何科学高效投放Facebook Ads

中午好家人们~今天Zoey来聊聊如何科学高效投放Facebook Ads~ 一、定义目标受众 在开始广告投放之前&#xff0c;需要明确定义你的目标受众。你可以根据受众的年龄、性别、兴趣、行为以及他们所在的地理位置等信息来确定目标受众。这样有助于创建精准的广告&#xff0c;并确保广…

PieCloudDB Database Flink Connector:让数据流动起来

面对客户环境中长期运行的各种类型的传统数据库&#xff0c;如何优雅地设计数据迁移的方案&#xff0c;既能灵活地应对各种数据导入场景和多源异构数据库&#xff0c;又能满足客户对数据导入结果的准确性、一致性、实时性的要求&#xff0c;让客户平滑地迁移到 PieCloudDB 数据…

降重工具:AI辅助下的论文查重率优化

确保论文的原创性和学术诚信是每位学生毕业的关键步骤&#xff0c;而论文查重和降重在此过程中起着至关重要的作用。 传统的论文查重通常依赖于查重软件和个人复查&#xff0c;而降重则涉及改写、同义词替换、内容的扩展与深化以及正确引用等方法&#xff0c;这些步骤不仅耗时…

MySQL——索引失效的10种情况

MySQL中提高性能的一个最有效的方式是对数据表设计合理的索引。索引提供了高效访问数据的方法&#xff0c;并且加快查询速度&#xff0c;因此索引对查询的速度有着至关重要的影响。 使用索引可以快速定位表中的某条记录&#xff0c;从而提高数据库查询的速度&#xff0c;提高数…

功能强大且专业的PDF转换软件PDF Shaper Professional 14.2

PDF Shaper Professional是一款适用于Windows的程序&#xff0c;可让您在计算机上处理PDF文件。 要开始使用PDF Shaper Professional&#xff0c;您需要在Windows计算机上下载并安装该程序。您还应该有合适的驱动程序和编解码器来处理计算机上的文本和图形。 安装程序后&#…

常用运维工具之 WGCLOUD(国产软件)介绍

WGCLOUD是一款免费开源的运维监控软件&#xff0c;轻量高效&#xff0c;部署方便&#xff0c;上手简单&#xff0c;界面简单流畅 WGCLOUD是国产运维软件&#xff0c;可以适配大部分的信创环境&#xff0c;比如麒麟、统信等操作系统 WGCLOUD具体支持监控的操作系统如下&#x…

【全开源】小区入户安检系统(FastAdmin + Uni-APP)

守护家的每一道防线 一款基于FastAdmin Uni-APP开发的小区入户安检系统(前端可发布为小程序、H5、App)。可针对不同行业自定义安检项目&#xff0c;线下安检&#xff0c;线上留存&#xff08;安检拍照/录像&#xff09;&#xff0c;提高安检人员安检效率。 一、引言&#xff…

php反序列化入门

一&#xff0c;php面向对象。 1.面向对象&#xff1a; 以“对象”伪中心的编程思想&#xff0c;把要解决的问题分解成对象&#xff0c;简单理解为套用模版&#xff0c;注重结果。 2.面向过程&#xff1a; 以“整体事件”为中心的编程思想&#xff0c;把解决问题的步骤分析出…

美洽工作台3.0,全新发布!

美洽工作台3.0&#xff0c;全新发布 想要效率翻倍&#xff0c;就要一步到位&#xff01; 工作台 3.0&#xff0c;为效率而生 1. 更丰富的外观选择&#xff0c;让界面焕然一新&#xff0c;新增导航主题色选择&#xff0c;深色 Dark、浅色 Light 随意切换 2. 自定义你的专属导…

基于STM32的位置速度环PID控制伺服电机转动位置及程序说明

PID控制原理 PID控制原理是一种广泛应用于工业自动化和其他领域的控制算法。PID控制器的名字来源于其三个主要组成部分&#xff1a;比例&#xff08;Proportional&#xff09;、积分&#xff08;Integral&#xff09;和微分&#xff08;Derivative&#xff09;。PID控制器实现…

C++数组实现推箱子游戏

前言 我是三天打鱼两天晒网的闲人,今天跟着课程视频学习c的数组的运用. 准备好游戏用到的图片资源 代码逻辑实现 #include<iostream> #include<graphics.h> #include<string> #include<conio.h>using namespace std;//设置画布大小 #define SCREEN…