Dijkstra算法
一 Dijkstra算法
Dijkstra算法解决了有向图G=(V, E)上带权的单源最短路径问题,但要求所有边的权值非负。Dijkstra算法采用贪心策略。
最短路径的最优子结构:最短路径的子路径是最短路径。
从源顶点s出发,可以确认唯一的顶点u1(除非存在顶点v1,其weight(s,v1)=weight(s,u1)),使它的前趋为s,且s->u为最短路径,此时weight(s,u1)在s的各临边中最小。与s相邻的其他顶点,总会通过松弛方法得出最短路径。
除了邻接表外还有3个数组,当前最短距离distance,前趋顶点pre,是否已访问过顶点(即已确定好最短路径的顶点)visited。
假设单源顶点为srcIdx,则Dijkstra算法步骤如下:
① 初始化:distance设置为INT_MAX,pre设置为-1,visited设为false,distance[srcIdx]=0
② 循环VertexNum次:每次从distance中取出visited为false,且distance最短的minIdx,此时将visited[minIdx]变为true,并通过松弛的方法确认minIdx的后续各顶点的最短路径上界。每次循环可以确定源顶点srcIdx到某个顶点的最短路径。
③ 递归打印源顶点srcIdx到各顶点的最短路径,及最短长度。
核心代码:
1 void initializeSingleSource(int srcIdx, int VertexNum, int *distance, int *pre, bool *visited) 2 { 3 for (int i=0;i<VertexNum;i++) 4 { 5 distance[i] = INT_MAX; 6 pre[i] = -1; 7 visited[i] = false; 8 } 9 distance[srcIdx] = 0; 10 } 11 12 int extractMin(bool *visited, int *distance) 13 {// 获取当前距离最短的顶点索引 14 int distanceMax = INT_MAX; 15 int minIdx = -1; 16 for (int j=0;j<VertexNum;j++) 17 { 18 if (visited[j] == false && distance[j] < distanceMax) 19 { 20 distanceMax = distance[j]; 21 minIdx = j; 22 } 23 } 24 return minIdx; 25 } 26 27 void relax(int u, int v, int w, int *d, int *pre) 28 { 29 if (d[v] > d[u] + w) 30 { 31 d[v] = d[u] + w; 32 pre[v] = u; 33 } 34 } 35 36 void visitPre(int idx, int *pre, int distance) 37 { 38 if (pre[idx] != -1) 39 { 40 visitPre(pre[idx], pre, distance); 41 cout << "-->" << idx; 42 } 43 else 44 cout << idx << "(" << distance << ")"; 45 } 46 47 void printShortestPath(int *pre, int *distance) 48 { 49 cout << "打印最短路径及最短长度:" << endl; 50 for (int i=0;i<VertexNum;i++) 51 { 52 visitPre(i, pre, distance[i]); 53 cout << endl; 54 } 55 } 56 57 void dijkstra(Vertex *graph, int srcIdx, int VertexNum) 58 { 59 int *distance = new int[VertexNum]; 60 int *pre = new int[VertexNum]; 61 bool *visited = new bool[VertexNum]; 62 63 initializeSingleSource(srcIdx, VertexNum, distance, pre, visited); 64 65 for (int i=0;i<VertexNum;i++) 66 { 67 int minIdx = extractMin(visited, distance); 68 visited[minIdx] = true; 69 70 Vertex *vertex = &graph[minIdx]; 71 while (vertex->next) 72 {// 访问该顶点的所有后续顶点,确立当前的最短路径 73 int vIdx = vertex->next->idx; 74 if (visited[vIdx] == false) 75 relax(minIdx, vIdx, vertex->next->weight, distance, pre); 76 vertex = vertex->next; 77 } 78 } 79 80 printShortestPath(pre, distance); 81 }
截图:
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】博客园携手 AI 驱动开发工具商 Chat2DB 推出联合终身会员
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 对象分配(Alloc)底层原理浅谈
· 聊一聊 C#异步 任务延续的三种底层玩法
· 敏捷开发:如何高效开每日站会
· 为什么 .NET8线程池 容易引发线程饥饿
· golang自带的死锁检测并非银弹
· 一个适用于 .NET 的开源整洁架构项目模板
· API 风格选对了,文档写好了,项目就成功了一半!
· 【开源】C#上位机必备高效数据转换助手
· .NET 9.0 使用 Vulkan API 编写跨平台图形应用
· .NET 依赖注入中的 Captive Dependency