狄克斯特拉算法求最短路径

cpp 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#define MAX 100
#define NO 0xFFFFFF

typedef struct Graph
{
	int arcnum;
	int vexnum;
	char vextex[MAX][20];
	int matrix[MAX][MAX];
}Graph;
void print_graph(Graph* g)
{
	for (int i = 0; i < g->vexnum; i++)
	{
		printf("\t%s", g->vextex[i]);
	}
	printf("\n");
	for (int i = 0; i < g->vexnum; i++)
	{
		printf("%s\t", g->vextex[i]);
		for (int j = 0; j < g->vexnum; j++)
		{
			if (g->matrix[i][j] == NO)
			{
				printf("NO\t");
			}
			else
			{
				printf("%d\t", g->matrix[i][j]);
			}
		}
		printf("\n");
	}
}
//狄克斯特拉算法
void graph_dijkstra(Graph* g, int in, int dist[])
//in为起始要求的到其他点最短路径的点,dist数组用于记录到该点的距离(如果有更短的则在上面更新) 
{
	int flag[MAX];//成功获取路径的标记
	int path[MAX];//记录每个节点的前驱节点
	for (int i = 0; i < g->vexnum; i++)
	{
		flag[i] = 0;	//将所有顶点都置为未访问 				
		dist[i] = g->matrix[in][i]; //获取起始顶点到其它所有顶点的权值   	
		path[i] = in;			//先将所有结点的前驱结点都设为初始的in顶点
	}
	flag[in] = 1;  //起始节点置为已访问
	dist[in] = 0;  //起始顶点自己到自己的距离为0
	int min;
	int j;//用于控制循环
	int k;//用于记录边权值最小终点的下标
	for (int i = 1; i < g->vexnum; i++)
	{
		//找最小的
		min = NO;
		for (j = 1; j < g->vexnum; j++)//注意j要从1开始,因为起始点下标为0 
		{
			//当该节点未被访问(或者说该边权值未被加入到最短路径中)且该点对应边权值已被加入到dist数组中去 
			if (flag[j] == 0 && dist[j] < min)
			{
				min = dist[j]; //先找到与邻接点最小的权值并记录,且记录该结点
				k = j;
			}
			//最终找到最小的 
		}
		//加上原路径 小于直接路径 更新
		flag[k] = 1;               //将找到的邻接边权值最小的终点置为已访问 
		for (j = 1; j < g->vexnum; j++)
		{
			if (flag[j] == 0 && min + g->matrix[k][j] < dist[j])
			{
				//如果该结点没被访问且加上原路径小于直接路径则更新,将边权值加入到数组dist中
			//这个操作相当于开辟边,让下一次的循环是一个有边的状态,可用于比对然后加入将最短权值点置为已访问 
				dist[j] = min + g->matrix[k][j];
				path[j] = k; //将其前驱节点置为k
			}
		}
	}
	//使用数据结构栈来打印最短路径
	for (int i = 1; i < g->vexnum; i++)
	{
		printf("从%s到%s的最短路径长度为%2d\t",g->vextex[in],g->vextex[i],dist[i]);
		//输出路径
		printf("路径为:%s\t",g->vextex[in]);
		char stack[MAX][20];
		int top = -1;
		int prev = path[i];//prev是该终点的前驱节点
		while (prev != in)
		{
			strcpy_s(stack[++top], 20, g->vextex[prev]);
			prev = path[prev];
		}
		while (top != -1)
		{
			printf("%s\t",stack[top--]);
		}
		printf("%s\n",g->vextex[i]);//最后再将终点输出来
	}
}

int main(int argc, char* argv[])
{
	Graph g =
	{ 9,6,{"A","B","C","D","E","F"},
	  {
		  {NO,NO,5,30,NO,NO},
		  {2,NO,NO,NO,8,NO},
		  {NO,15,NO,NO,NO,7},
		  {NO,NO,NO,NO,NO,NO},
		  {NO,NO,NO,4,NO,NO},
		  {NO,NO,NO,10,18}
	  }
	};
	print_graph(&g);
	int in = 0;
	int dist[MAX] = { 0 };
	graph_dijkstra(&g, in, dist);
	return 0;

}
相关推荐
嘉友4 分钟前
Redis zset数据结构以及时间复杂度总结(源码)
数据结构·数据库·redis·后端
无难事者若执8 分钟前
新手村:逻辑回归-理解03:逻辑回归中的最大似然函数
算法·机器学习·逻辑回归
IT从业者张某某24 分钟前
机器学习-04-分类算法-03KNN算法案例
算法·机器学习·分类
chen_song_28 分钟前
WebRTC的ICE之TURN协议的交互流程中继转发Relay媒体数据的turnserver的测试
算法·音视频·webrtc·交互·媒体
蒙奇D索大44 分钟前
【数据结构】图解图论:度、路径、连通性,五大概念一网打尽
数据结构·考研·算法·图论·改行学it
uhakadotcom1 小时前
2025年春招:如何使用DeepSeek + 豆包优化简历,轻松敲开心仪公司的大门
算法·面试·github
小白狮ww1 小时前
Retinex 算法 + MATLAB 软件,高效率完成图像去雾处理
开发语言·人工智能·算法·matlab·自然语言处理·图像识别·去雾处理
trust Tomorrow2 小时前
每日一题-力扣-2278. 字母在字符串中的百分比 0331
算法·leetcode
Lecea_L2 小时前
你能在K步内赚最多的钱吗?用Java解锁最大路径收益算法(含AI场景分析)
java·人工智能·算法
Tony882 小时前
热题100 - 394. 字符串解码
java·算法