Go语言实战:图的邻接表表示法实现详解

本文是《Go语言100个实战案例》系列中的一篇,聚焦图的邻接表表示法,结合Go语言进行数据结构与算法的实践。适合初学者以及有一定基础的开发者学习图结构在实际工程中的应用。


一、什么是图的邻接表表示?

在图(Graph)的表示方法中,邻接表(Adjacency List) 是一种常用的结构。它适用于稀疏图,即边数远少于点数平方的图。邻接表使用链表或切片来保存每个顶点的相邻顶点,相较邻接矩阵节省大量空间。

邻接表的核心思想:

  • • 每个顶点对应一个链表(或切片),存储与该顶点相邻的边(即相邻的顶点)。
  • • 可以用于有向图和无向图。

二、Go语言中图的邻接表结构设计

我们使用 Go 的 map 和 slice 数据结构实现邻接表。下面是一个基础结构:

go 复制代码
package main

import "fmt"

// Graph 表示图结构(使用邻接表)
type Graph struct {
    vertices map[string][]string
    isDirected bool
}

// NewGraph 构造函数
func NewGraph(isDirected bool) *Graph {
    return &Graph{
        vertices:   make(map[string][]string),
        isDirected: isDirected,
    }
}

// AddVertex 添加顶点
func (g *Graph) AddVertex(v string) {
    if _, exists := g.vertices[v]; !exists {
        g.vertices[v] = []string{}
    }
}

// AddEdge 添加边
func (g *Graph) AddEdge(from, to string) {
    g.AddVertex(from)
    g.AddVertex(to)

    g.vertices[from] = append(g.vertices[from], to)
    if !g.isDirected {
        g.vertices[to] = append(g.vertices[to], from)
    }
}

// PrintGraph 输出邻接表
func (g *Graph) PrintGraph() {
    for vertex, neighbors := range g.vertices {
        fmt.Printf("%s -> %v\n", vertex, neighbors)
    }
}

三、使用示例:构建一个简单的无向图

css 复制代码
func main() {
    graph := NewGraph(false) // false 表示无向图

    graph.AddEdge("A", "B")
    graph.AddEdge("A", "C")
    graph.AddEdge("B", "D")
    graph.AddEdge("C", "D")
    graph.AddEdge("D", "E")

    graph.PrintGraph()
}

输出:

css 复制代码
A -> [B C]
B -> [A D]
C -> [A D]
D -> [B C E]
E -> [D]

四、支持有向图

我们只需要在构造图的时候传入 true,表示是有向图:

css 复制代码
graph := NewGraph(true) // 有向图
graph.AddEdge("A", "B")
graph.AddEdge("A", "C")
graph.PrintGraph()

输出:

css 复制代码
A -> [B C]
B -> []
C -> []

五、扩展思路

为了适用于更多实际场景,我们可以拓展该邻接表:

  • • 支持权重:将 []string 改为 []Edge,其中 Edge 结构体包含 ToWeight 字段。
  • • 实现图遍历:如 BFS、DFS 算法。
  • • 实现图算法:如 Dijkstra 最短路径、拓扑排序、最小生成树等。
  • • 可视化:输出 Graphviz DOT 格式可视化图结构。

六、小结

邻接表是图结构中非常高效且实用的一种表示方式,尤其适用于稀疏图。在Go语言中,我们可以利用 map 和 slice 快速构建邻接表,为实现更复杂的图算法打下基础。

相关推荐
许苑向上24 分钟前
Spring Boot 自动装配底层源码实现详解
java·spring boot·后端
再学一点就睡1 小时前
手写 Promise 静态方法:从原理到实现
前端·javascript·面试
再学一点就睡1 小时前
前端必会:Promise 全解析,从原理到实战
前端·javascript·面试
超级小忍3 小时前
深入浅出:在 Spring Boot 中构建实时应用 - 全面掌握 WebSocket
spring boot·后端·websocket
没有bug.的程序员3 小时前
《Spring Security源码深度剖析:Filter链与权限控制模型》
java·后端·spring·security·filter·权限控制
OEC小胖胖3 小时前
性能优化(一):时间分片(Time Slicing):让你的应用在高负载下“永不卡顿”的秘密
前端·javascript·性能优化·web
小小李程序员4 小时前
JSON.parse解析大整数踩坑
开发语言·javascript·json
无责任此方_修行中4 小时前
不止是 AI 热潮:AWS 2025 技术峰会带给我的思考
后端·架构·aws
宋辰月4 小时前
Vue2-VueRouter
开发语言·前端·javascript
lang201509284 小时前
Apache Ignite 与 Spring Boot 集成
spring boot·后端·apache·ignite