mirror of
https://github.com/krahets/hello-algo.git
synced 2024-12-25 13:16:29 +08:00
e720aa2d24
* Sync recent changes to the revised Word. * Revised the preface chapter * Revised the introduction chapter * Revised the computation complexity chapter * Revised the chapter data structure * Revised the chapter array and linked list * Revised the chapter stack and queue * Revised the chapter hashing * Revised the chapter tree * Revised the chapter heap * Revised the chapter graph * Revised the chapter searching * Reivised the sorting chapter * Revised the divide and conquer chapter * Revised the chapter backtacking * Revised the DP chapter * Revised the greedy chapter * Revised the appendix chapter * Revised the preface chapter doubly * Revised the figures
100 lines
2.2 KiB
Go
100 lines
2.2 KiB
Go
// File: graph_adjacency_list.go
|
||
// Created Time: 2023-01-31
|
||
// Author: Reanon (793584285@qq.com)
|
||
|
||
package chapter_graph
|
||
|
||
import (
|
||
"fmt"
|
||
"strconv"
|
||
"strings"
|
||
|
||
. "github.com/krahets/hello-algo/pkg"
|
||
)
|
||
|
||
/* 基于邻接表实现的无向图类 */
|
||
type graphAdjList struct {
|
||
// 邻接表,key:顶点,value:该顶点的所有邻接顶点
|
||
adjList map[Vertex][]Vertex
|
||
}
|
||
|
||
/* 构造函数 */
|
||
func newGraphAdjList(edges [][]Vertex) *graphAdjList {
|
||
g := &graphAdjList{
|
||
adjList: make(map[Vertex][]Vertex),
|
||
}
|
||
// 添加所有顶点和边
|
||
for _, edge := range edges {
|
||
g.addVertex(edge[0])
|
||
g.addVertex(edge[1])
|
||
g.addEdge(edge[0], edge[1])
|
||
}
|
||
return g
|
||
}
|
||
|
||
/* 获取顶点数量 */
|
||
func (g *graphAdjList) size() int {
|
||
return len(g.adjList)
|
||
}
|
||
|
||
/* 添加边 */
|
||
func (g *graphAdjList) addEdge(vet1 Vertex, vet2 Vertex) {
|
||
_, ok1 := g.adjList[vet1]
|
||
_, ok2 := g.adjList[vet2]
|
||
if !ok1 || !ok2 || vet1 == vet2 {
|
||
panic("error")
|
||
}
|
||
// 添加边 vet1 - vet2, 添加匿名 struct{},
|
||
g.adjList[vet1] = append(g.adjList[vet1], vet2)
|
||
g.adjList[vet2] = append(g.adjList[vet2], vet1)
|
||
}
|
||
|
||
/* 删除边 */
|
||
func (g *graphAdjList) removeEdge(vet1 Vertex, vet2 Vertex) {
|
||
_, ok1 := g.adjList[vet1]
|
||
_, ok2 := g.adjList[vet2]
|
||
if !ok1 || !ok2 || vet1 == vet2 {
|
||
panic("error")
|
||
}
|
||
// 删除边 vet1 - vet2
|
||
g.adjList[vet1] = DeleteSliceElms(g.adjList[vet1], vet2)
|
||
g.adjList[vet2] = DeleteSliceElms(g.adjList[vet2], vet1)
|
||
}
|
||
|
||
/* 添加顶点 */
|
||
func (g *graphAdjList) addVertex(vet Vertex) {
|
||
_, ok := g.adjList[vet]
|
||
if ok {
|
||
return
|
||
}
|
||
// 在邻接表中添加一个新链表
|
||
g.adjList[vet] = make([]Vertex, 0)
|
||
}
|
||
|
||
/* 删除顶点 */
|
||
func (g *graphAdjList) removeVertex(vet Vertex) {
|
||
_, ok := g.adjList[vet]
|
||
if !ok {
|
||
panic("error")
|
||
}
|
||
// 在邻接表中删除顶点 vet 对应的链表
|
||
delete(g.adjList, vet)
|
||
// 遍历其他顶点的链表,删除所有包含 vet 的边
|
||
for v, list := range g.adjList {
|
||
g.adjList[v] = DeleteSliceElms(list, vet)
|
||
}
|
||
}
|
||
|
||
/* 打印邻接表 */
|
||
func (g *graphAdjList) print() {
|
||
var builder strings.Builder
|
||
fmt.Printf("邻接表 = \n")
|
||
for k, v := range g.adjList {
|
||
builder.WriteString("\t\t" + strconv.Itoa(k.Val) + ": ")
|
||
for _, vet := range v {
|
||
builder.WriteString(strconv.Itoa(vet.Val) + " ")
|
||
}
|
||
fmt.Println(builder.String())
|
||
builder.Reset()
|
||
}
|
||
}
|