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 快速构建邻接表,为实现更复杂的图算法打下基础。

相关推荐
LunarCod10 分钟前
Hexo搭建/部署个人博客教程
后端·hexo·个人博客·vercel
克里斯蒂亚诺更新1 小时前
微信小程序app.js中每30秒调用一次wx.getLocation
javascript·微信小程序·小程序
IT_陈寒1 小时前
Vue 3.4 实战:这7个Composition API技巧让我的开发效率飙升50%
前端·人工智能·后端
鄃鳕1 小时前
C++坑系列,C++ std::atomic 拷贝构造函数问题分析与解决方案
java·javascript·c++
风雨同舟的代码笔记2 小时前
ThreadLocal的使用以及源码分析
后端
Never_Satisfied2 小时前
在JavaScript / HTML中,实现`<iframe>` 自适应高度
开发语言·javascript·html
brzhang2 小时前
把网页的“好句子”都装进侧边栏:我做了个叫 Markbox 的收藏器,开源!
前端·后端·架构
猎豹奕叔4 小时前
JD到家商品系统架构设计演进
后端
阑梦清川4 小时前
深入理解动静态库和ELF文件格式
后端
猎豹奕叔4 小时前
面试官:类中两个方法加同步锁,多线程能同时访问吗?
后端