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

目录

  • 最短路径 (*)
    • 迪杰斯特拉算法(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 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

导弹研究中常用坐标系及坐标系之间的变换

在导弹飞行控制过程中&#xff0c;需要时刻掌握导弹的飞行状态 &#xff08;速度、位置、姿态角等&#xff09;&#xff0c;这就有赖于描述导弹飞行状态的坐标系。除了大地坐标系和地心大地直角坐标系外&#xff0c;导弹常用的坐标系还有很多&#xff0c;合理而恰当地选择参考系…

【LangChain系列】第二篇:文档拆分简介及实践

在上一篇博客中&#xff0c;我们学习了如何使用LangChain的文档加载器将文档加载为标准格式。加载文档后&#xff0c;下一步是将它们拆分为更小的块。这个过程乍一看似乎很简单&#xff0c;但有一些微妙之处和重要的考虑因素会显着影响下游任务的性能和准确性。 一、为什么文档…

qcom 平台系统签名流程

security boot 平台的东东&#xff0c;oem 可定制的功能有限&#xff0c;只能参考平台文档&#xff0c;可以在高通的网站上搜索&#xff1a;Secure Boot Enablement&#xff0c;然后找对应平台的文档xxx-Secure Boot Enablement User Guide, step by step 操作即可 开机校验流…

【人工智能】第五部分:ChatGPT的实际应用案例和未来发展方向

人不走空 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌赋&#xff1a;斯是陋室&#xff0c;惟吾德馨 目录 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌…

【成品设计】基于STM32的体温监控系统的设计与实现

《基于STM32的体温监控系统的设计与实现》 整体功能&#xff1a; 体温监控系统采用STM32F103VET6单片机为主控&#xff0c;在此基础上移植了一款国产嵌入式操作系统RT-thread进行开发设计的。 该系统的主要工作逻辑为&#xff1a;管理员可先将相关人员的指纹、ID等数据录入系…

AC/DC电源模块的效率如何?

BOSHIDA AC/DC电源模块的效率如何&#xff1f; AC/DC电源模块是一种将交流电转换为直流电的装置&#xff0c;常用于电子设备的电源部分。其效率是指输入电功率与输出电功率的比值&#xff0c;通常以百分比表示。AC/DC电源模块的效率主要取决于以下几个因素&#xff1a;开关频…

EE trade:如何在A股市场中有效设定止盈止损点

A股市场充满机遇和风险&#xff0c;很多投资者在这里实现了财富增长&#xff0c;也有投资者在这里遭受损失。如何在波动性较大的市场中&#xff0c;控制风险&#xff0c;保护利润和本金?止盈止损是关键。 什么是止盈止损? 止盈止损是指在交易中&#xff0c;根据预先设定的条…

酱菜产业:传承美味,点亮生活

酱菜&#xff0c;这道深受人们喜爱的传统美食&#xff0c;以其独特的风味和营养价值&#xff0c;点亮了我们的日常生活。酱菜产业作为美食文化的重要组成部分&#xff0c;正以其独特的魅力&#xff0c;吸引着越来越多的消费者。 酱菜产业的赵总说&#xff1a;酱菜的制作过程&am…

深入分析 Flink SQL 工作机制

摘要&#xff1a;本文整理自 Flink Forward 2020 全球在线会议中文精华版&#xff0c;由 Apache Flink PMC 伍翀&#xff08;云邪&#xff09;分享&#xff0c;社区志愿者陈婧敏&#xff08;清樾&#xff09;整理。旨在帮助大家更好地理解 Flink SQL 引擎的工作原理。文章主要分…

电商API商品数据采集接口||助力电商企业采集商品大数据提高开发效率

提高开发效率&#xff1a;电商API接口允许不同的应用程序之间高效地进行交互&#xff0c;节省了大量的人力物力成本&#xff0c;使得开发者可以将更多时间和精力集中于自身的核心业务。 增加数据安全性&#xff1a;通过对数据进行安全加密&#xff0c;API接口实现了对数据的保护…

java自学阶段二:JavaWeb开发45(git学习)

目录&#xff1a; 学习目标git的使用&#xff08;工作流程、常用命令、idea集成&#xff09; 一、学习目标&#xff1a; 了解Git基本概念能够了解git的工作流程能够使用Git常用命令熟悉Git代码托管服务能够使用idea操作git 二、git的使用 1&#xff09;git的概念&#xff1…

Oracle 19c OCM认证

Oracle OCM介绍 Oracle Certified Master (OCM) -Oracle认证大师&#xff0c;是Oracle认证的最高级别&#xff0c;是对数据库从业人员的技术、知识和操作技能的最高级别的认可&#xff0c;IT界顶级认证之一。Oracle OCM是解决最困难的技术难题和最复杂的系统故障的最佳Oracle专…

凡尔码搭建设备巡检系统数字化管理平台

一、搭建过程概述 利用凡尔码搭建设备巡检的数字化管理平台&#xff0c;首先需要对凡尔码平台有深入的了解&#xff0c;明确其提供的核心功能和特性&#xff0c;以及如何在设备巡检领域发挥其优势。接着&#xff0c;通过系统规划、组件配置、数据录入和表单创建等步骤&#xff…

短剧小程序App系统源码:打造个性化追剧体验

随着数字媒体的迅猛发展&#xff0c;短剧作为一种新兴的娱乐形式&#xff0c;越来越受到广大观众的喜爱。为了满足用户对短剧内容的个性化需求&#xff0c;短剧小程序App系统应运而生。本文将深入探讨短剧App源码的核心功能&#xff0c;以及如何通过多语言支持和国际支付等技术…

形如SyntaxError: EOL while scanning string literal,以红色波浪线形式在Pycharm下出现

背景&#xff1a; 新手在学习Python时可能会出现如下图所示的报错 下面分情况教大家如何解决 过程&#xff1a; 问题概述&#xff1a; 简单来说就是你单引号、双引号、三引号写的时候末尾注意要和前面写的匹配。 具体如下 """ 编辑器报错&#xff1a;Synt…

在Windows中使用svn的命令行

windows下使用svn命令行_svn命令行工具在哪里-CSDN博客 先下载命令行工具 再进行配置 set SVN_CMD_HOMEC:\Users\admin\Desktop\Apache-Subversion-1.14.0\bin(你的安装路径) set path%path%;%SVN_CMD_HOME% svn help查看svn版本 命令行查看svn版本--真实有效_svn 版本查看…

Java Web学习笔记4——HTML、CSS

HTML&#xff1a; HTML&#xff1a;超文本标记语言。 超文本&#xff1a;超越了文本的限制&#xff0c;比普通文本更强大。除了文字信息&#xff0c;还可以定义图片、音频、视频等内容。 标记语言&#xff1a;有标签构成的语言。 HTML标签都是预定义好的&#xff0c;例如&a…

【全开源】小区物业收费管理系统小程序(FastAdmin+UniApp)

便捷生活新选择 一款基于FastAdminUniApp开发的一款物业收费管理小程序。包含房产管理、收费标准、家属管理、抄表管理、在线缴费、业主公告、统计报表、业主投票、可视化大屏等功能。为物业量身打造的小区收费管理系统&#xff0c;贴合物业工作场景&#xff0c;轻松提高物业费…

Python实现PPT表格的编写包含新建修改插图(收藏备用)

自动创建一个ppt文件并创建好表格 代码要用到pptx库 pip install python-pptx 创建含有表格的ppt文件代码&#xff1a; from pptx import Presentation from pptx.util import Inches# 创建一个PPT对象 ppt Presentation()# 添加一个幻灯片 slide ppt.slides.add_slide(p…

【C++】优先级队列介绍与模拟实现

&#x1f49e;&#x1f49e; 前言 hello hello~ &#xff0c;这里是大耳朵土土垚~&#x1f496;&#x1f496; &#xff0c;欢迎大家点赞&#x1f973;&#x1f973;关注&#x1f4a5;&#x1f4a5;收藏&#x1f339;&#x1f339;&#x1f339; &#x1f4a5;个人主页&#x…