0

0

Go语言中Dijkstra算法的最短路径重建教程

花韻仙語

花韻仙語

发布时间:2025-11-09 13:57:21

|

618人浏览过

|

来源于php中文网

原创

go语言中dijkstra算法的最短路径重建教程

本文详细介绍了如何在Go语言实现的Dijkstra算法中,不仅计算出源点到各顶点的最短距离,还能有效地重建并打印出实际的最短路径。核心方法是在图的顶点结构中引入一个前驱(Prev)指针,当算法更新最短距离时同步记录路径上的前一个顶点,从而在算法结束后通过回溯这些指针来逆向构建出完整的路径。

在图论算法中,Dijkstra算法是解决单源最短路径问题的经典方法。它能够高效地计算出从起始顶点到图中所有其他顶点的最短距离。然而,在许多实际应用场景中,仅仅知道最短距离是不够的,我们还需要知道构成这条最短路径的具体顶点序列。本教程将基于一个Go语言实现的Dijkstra算法示例,详细讲解如何对其进行改造,以实现最短路径的重建与打印。

理解最短路径重建的核心原理

Dijkstra算法通过不断松弛边来更新从源点到各个顶点的最短距离。当算法发现一条从源点到某个目标顶点的新路径比已知路径更短时,它会更新该目标顶点的最短距离。为了重建路径,我们需要的正是这种“更新”操作:每当一个顶点的最短距离被更新时,我们同时记录是哪个顶点导致了这次更新。这个“导致更新”的顶点就是目标顶点在最短路径上的直接前驱。

通过为每个顶点维护一个指向其前驱的指针,算法结束后,我们可以从目标顶点开始,沿着这些前驱指针反向回溯,直到到达源点,从而得到一条从源点到目标顶点的最短路径(逆序)。

立即学习go语言免费学习笔记(深入)”;

修改顶点结构以支持路径记录

首先,我们需要修改表示图的顶点结构,为其添加一个用于存储前驱顶点的字段。

type Vertex struct {
    Id      string
    Visited bool
    AdjEdge []*Edge
    Prev    *Vertex // 新增:指向最短路径上的前一个顶点
}

在这里,Prev *Vertex 字段将用来存储当前顶点在最短路径上的直接前驱。初始时,所有顶点的 Prev 字段应为 nil。

修改Dijkstra算法以记录前驱

接下来,我们需要在Dijkstra算法的核心逻辑中,即当发现并更新一个更短路径时,同步设置 Prev 字段。在原始的 CalculateD 函数中,更新最短距离的逻辑如下:

Kacha
Kacha

KaCha是一款革命性的AI写真工具,用AI技术将照片变成杰作!

下载
if D[edge.Destination] > D[edge.Source]+edge.Weight {
    D[edge.Destination] = D[edge.Source] + edge.Weight
    // ... 在这里添加记录前驱的逻辑
}

修改后的 CalculateD 函数(或任何执行松弛操作的地方)应包含以下逻辑:

const MAXWEIGHT = 1000000

type MinDistanceFromSource map[*Vertex]int

// Dijks 函数保持不变,或根据需要初始化所有Prev为nil
func (G *Graph) Dijks(StartSource, TargetSource *Vertex) MinDistanceFromSource {
  D := make(MinDistanceFromSource)
  for _, vertex := range G.VertexArray {
    D[vertex] = MAXWEIGHT
    vertex.Prev = nil // 初始化Prev指针
  }
  D[StartSource] = 0
  StartSource.Prev = nil // 源点没有前驱

  // 初始边的处理,同样需要设置Prev
  for edge := range StartSource.GetAdEdg() {
    if D[edge.Destination] > edge.Weight { // 确保是更短路径才更新
        D[edge.Destination] = edge.Weight
        edge.Destination.Prev = StartSource // 设置前驱
    }
  }
  CalculateD(StartSource, TargetSource, D) // 递归调用可能需要调整为迭代
  return D
}

// CalculateD 函数应调整为迭代式,这里仅展示关键的更新逻辑
// 原始的递归实现可能存在栈溢出风险,且不完全符合Dijkstra的典型迭代结构。
// 假设这是在Dijkstra主循环内部的松弛操作
func CalculateD(currentVertex *Vertex, D MinDistanceFromSource) {
    // 遍历当前顶点的所有邻接边
    for edge := range currentVertex.GetAdEdg() {
        // 如果通过当前顶点到达目标顶点的路径更短
        if D[edge.Destination] > D[currentVertex]+edge.Weight {
            D[edge.Destination] = D[currentVertex] + edge.Weight
            edge.Destination.Prev = currentVertex // 关键:更新前驱指针
        }
    }
    // 注意:一个完整的Dijkstra算法会有一个优先队列来选择下一个要处理的顶点
    // 这里的CalculateD片段仅展示了更新前驱的核心逻辑
}

重要提示: 原始代码中的 CalculateD 函数是一个递归实现,这在处理大型图时可能导致溢出,并且它不完全符合Dijkstra算法通常的迭代式实现(使用优先队列)。为了清晰地展示路径重建,我们假设上述 CalculateD 片段是Dijkstra主循环中松弛操作的一部分。在标准的Dijkstra实现中,你会有一个循环,每次从一个未访问的顶点中选择距离源点最近的那个,然后遍历其所有邻接边进行松弛。

重建并打印最短路径

Dijkstra算法执行完毕后,每个顶点(如果可达)的 Prev 字段都将指向其在最短路径上的前驱。要打印从源点到特定目标顶点的路径,我们只需从目标顶点开始,沿着 Prev 指针反向回溯,直到遇到 nil (通常是源点)。

以下是一个打印从源点到某个目标顶点路径的示例函数:

// PrintPath 从目标顶点回溯打印路径
func PrintPath(target *Vertex) {
    if target == nil {
        fmt.Println("目标顶点为空,无法打印路径。")
        return
    }

    path := []string{}
    current := target

    // 从目标顶点开始,反向回溯到源点
    for current != nil {
        path = append(path, current.Id)
        current = current.Prev
    }

    // 路径是逆序的,需要反转
    for i, j := 0, len(path)-1; i < j; i, j = i+1, j-1 {
        path[i], path[j] = path[j], path[i]
    }

    // 打印路径
    fmt.Print("最短路径: ")
    for i, id := range path {
        fmt.Print(id)
        if i < len(path)-1 {
            fmt.Print(" -> ")
        }
    }
    fmt.Println()
}

在你的主程序或结果展示部分,你可以这样调用:

// 假设 distmap1 是 Dijks 返回的结果,TargetSource 是你的目标顶点
// G.Dijks(StartSource, TargetSource) 应该返回一个包含Prev指针的Vertex结构
// ... 运行Dijkstra算法 ...

// 遍历所有顶点,打印它们的距离和路径
for vertex, distance := range distmap1 {
    fmt.Printf("从源点到 %s 的最短距离: %d\n", vertex.Id, distance)
    if distance != MAXWEIGHT { // 如果可达
        PrintPath(vertex)
    } else {
        fmt.Printf("从源点到 %s 不可达\n", vertex.Id)
    }
    fmt.Println("---")
}

注意事项与总结

  1. 初始化 Prev 指针: 在Dijkstra算法开始前,务必将所有顶点的 Prev 指针初始化为 nil。源点的 Prev 始终为 nil。
  2. Dijkstra算法的完整性: 提供的 CalculateD 递归片段并非一个完整的Dijkstra实现。一个健壮的Dijkstra通常使用一个优先队列来选择下一个要访问的顶点,以确保每次都从当前未访问顶点中选择距离源点最近的那个。在完整的迭代式Dijkstra实现中,路径重建逻辑同样是在松弛操作(即 if D[edge.Destination] > D[edge.Source]+edge.Weight)内部执行。
  3. 路径的逆序: 由于我们是从目标顶点回溯到源点,所以构建出的路径是逆序的。在打印之前,需要将其反转。
  4. 不可达顶点: 如果目标顶点不可达,其 Prev 链将不会回溯到源点,或者其距离仍为 MAXWEIGHT。在打印路径前进行检查是良好的实践。

通过引入一个简单的 Prev 指针并将其与Dijkstra算法的松弛操作同步,我们便能有效地从最短路径算法中提取出完整的路径信息。这使得Dijkstra算法在需要实际导航或路径展示的场景中更具实用价值。

相关专题

更多
edge是什么浏览器
edge是什么浏览器

Edge是一款由Microsoft开发的网页浏览器,是Windows 10操作系统中默认的浏览器,其目标是提供更快、更安全、更现代化的浏览器体验。本专题为大家提供edge浏览器相关的文章、下载、课程内容,供大家免费下载体验。

1255

2023.08.21

IE浏览器自动跳转EDGE如何恢复
IE浏览器自动跳转EDGE如何恢复

ie浏览器自动跳转edge的解决办法:1、更改默认浏览器设置;2、阻止edge浏览器的自动跳转;3、更改超链接的默认打开方式;4、禁用“快速网页查看器”;5、卸载edge浏览器;6、检查第三方插件或应用程序等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

373

2024.03.05

如何解决Edge打开但没有标题的问题
如何解决Edge打开但没有标题的问题

若 Microsoft Edge 浏览器打开后无标题(窗口空白或标题栏缺失),可尝试以下方法解决: 重启 Edge:关闭所有窗口,重新启动浏览器。 重置窗口布局:右击任务栏 Edge 图标 → 选择「最大化」或「还原」。 禁用扩展:进入 edge://extensions 临时关闭插件测试。 重置浏览器设置:前往 edge://settings/reset 恢复默认配置。 更新或重装 Edge:检查最新版本,或通过控制面板修复

832

2025.04.24

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

713

2023.08.22

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

371

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

563

2023.08.10

Go中Type关键字的用法
Go中Type关键字的用法

Go中Type关键字的用法有定义新的类型别名或者创建新的结构体类型。本专题为大家提供Go相关的文章、下载、课程内容,供大家免费下载体验。

233

2023.09.06

go怎么实现链表
go怎么实现链表

go通过定义一个节点结构体、定义一个链表结构体、定义一些方法来操作链表、实现一个方法来删除链表中的一个节点和实现一个方法来打印链表中的所有节点的方法实现链表。

442

2023.09.25

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

65

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.2万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号