几个概念:有向图、无向图、加权图、简单图、联通、联通分量、生成树、强连通分量、强联通图
图的存储:邻接矩阵(二维、一维)、邻接表
图的遍历:深搜、广搜
(无向加权图)最小生成树:kruskal算法过程及证明、prime算法过程
单源最短距离问题:dijkstra算法
深搜和回溯的区别:个人理解是深搜是图的一种遍历方式(树也属于图),回溯是只用于树

package main

import (
    "fmt"
    "math"
    "sort"
    "strconv"
)

func main() {
    vertexes := []string{"A", "B", "C", "D", "E", "F"}
    edges := []string{
        "AB,4", "AC,8", "AD,45", "AF,3",
        "BA,4", "BC,14", "BE,20", "BF,5",
        "CA,8", "CB,14", "CD,28", "CE,9",
        "DA,45", "DC,28", "DE,10",
        "EB,20", "EC,9", "EF,13", "ED,10",
        "FA,3", "FB,5", "FE,13",
    }
    g := NewGraph(vertexes, edges)

    g.DFS()
    g.BFS(0)
    g.Kruskal()
    g.Prime(0)
    g.Dijkstra(0)
}

type Vertex struct {
    Name      string
    FirstEdge *Edge
}

type Edge struct {
    V        int
    W        int
    Cost     int
    NextEdge *Edge
}

type Graph struct {
    VertexNum    int
    EdgeNum      int
    AdjacentList []Vertex
}

func NewGraph(vertexes []string, edges []string) Graph {
    g := Graph{}
    g.VertexNum = len(vertexes)
    g.EdgeNum = len(edges) / 2 // 无向图
    g.AdjacentList = make([]Vertex, len(vertexes))

    m := map[string]int{}
    for i, vertex := range vertexes {
        v := Vertex{
            Name: vertex,
        }
        g.AdjacentList[i] = v
        m[vertex] = i
    }

    for _, edgeStr := range edges {
        v := m[string(edgeStr[0])]
        w := m[string(edgeStr[1])]
        cost, _ := strconv.Atoi(edgeStr[3:])
        edge := Edge{
            V:        v,
            W:        w,
            Cost:     cost,
            NextEdge: g.AdjacentList[v].FirstEdge,
        }
        g.AdjacentList[v].FirstEdge = &edge
    }

    return g
}

func (g *Graph) DFS() {
    visited := map[int]bool{}
    for i, _ := range g.AdjacentList {
        g.dfs(i, visited)
    }
}

func (g *Graph) dfs(v int, visited map[int]bool) {
    if visited[v] {
        return
    }
    vertex := g.AdjacentList[v]
    visited[v] = true
    fmt.Println("先深遍历", vertex.Name)

    edge := vertex.FirstEdge
    for edge != nil {
        if visited[edge.W] {
            edge = edge.NextEdge
            continue
        }
        g.dfs(edge.W, visited)
    }
}

func (g *Graph) BFS(start int) {
    visited := map[int]bool{}

    var queue = []int{
        start,
    }

    i := 0
    for ; i < len(queue); i++ {
        k := queue[i]
        vertex := g.AdjacentList[k]
        if visited[k] {
            continue
        }
        fmt.Println("先广遍历", vertex.Name)
        visited[k] = true

        edge := vertex.FirstEdge
        for edge != nil {
            if !visited[edge.W] {
                queue = append(queue, edge.W)
            }
            edge = edge.NextEdge
        }
    }
}

func (g *Graph) Kruskal() (mst []Edge) {
    sortedEdges := g.sortEdges()
    mst = make([]Edge, g.VertexNum-1)

    i := 0
    l := 0
    selectedVertexes := map[int]bool{}

    for l < g.VertexNum-1 {
        edge := sortedEdges[i]
        if !(selectedVertexes[edge.V] && selectedVertexes[edge.W]) {
            selectedVertexes[edge.V] = true
            selectedVertexes[edge.W] = true
            mst[l] = edge
            l++
        }
        i++
    }

    fmt.Println("Kruskal结果")
    for _, edge := range mst {
        fmt.Println(g.AdjacentList[edge.V].Name+g.AdjacentList[edge.W].Name, edge.Cost)
    }

    return
}

func (g *Graph) sortEdges() []Edge {
    visited := map[int]bool{}

    edges := make([]Edge, g.EdgeNum)
    l := 0
    for i, vertex := range g.AdjacentList {
        edge := vertex.FirstEdge
        for edge != nil {
            if !visited[edge.W] {
                edges[l] = *edge
                l++
            }
            edge = edge.NextEdge
        }
        visited[i] = true
    }

    sort.Slice(edges, func(i, j int) bool {
        return edges[i].Cost < edges[j].Cost
    })

    return edges
}

func (g *Graph) Prime(start int) (mst []Edge) {
    fmt.Println("prime计算过程")
    selectedVertex := map[int]bool{start: true}
    unSelectedVertex := map[int]Edge{}

    lastSelectVertex := start
    for len(mst) < g.VertexNum-1 {
        var minCost = math.MaxInt64
        var minEdge Edge

        for i, _ := range g.AdjacentList {
            if selectedVertex[i] {
                continue
            }
            newCost, _ := g.getCost(lastSelectVertex, i)
            if edge, initialized := unSelectedVertex[i]; !initialized || newCost < edge.Cost {
                unSelectedVertex[i] = Edge{
                    V:    lastSelectVertex,
                    W:    i,
                    Cost: newCost,
                }
            }

            if unSelectedVertex[i].Cost < minCost {
                minCost = unSelectedVertex[i].Cost
                minEdge = unSelectedVertex[i]
            }
        }

        fmt.Println()
        for i, edge := range unSelectedVertex {
            if selectedVertex[edge.W] {
                continue
            }
            fmt.Println(i, g.AdjacentList[edge.V].Name+g.AdjacentList[edge.W].Name, edge.Cost)
        }
        fmt.Println("选择最小边: ", 
g.AdjacentList[minEdge.V].Name+g.AdjacentList[minEdge.W].Name, minEdge.Cost)

        mst = append(mst, minEdge)
        lastSelectVertex = minEdge.W
        selectedVertex[minEdge.W] = true
    }

    fmt.Println("最终结果")
    for i, edge := range mst {
        fmt.Println(i, g.AdjacentList[edge.V].Name+g.AdjacentList[edge.W].Name, edge.Cost)
    }

    return
}

func (g *Graph) getVertex(i int) Vertex {
    return g.AdjacentList[i]
}

func (g *Graph) getCost(from int, to int) (cost int, isInfinity bool) {
    startVertex := g.AdjacentList[from]
    edge := startVertex.FirstEdge
    for edge != nil {
        if edge.W == to {
            return edge.Cost, false
        }
        edge = edge.NextEdge
    }
    return math.MaxInt64, true
}

func (g *Graph) Dijkstra(start int) {
    fmt.Println("dijkstra计算过程")
    type vertexRecord struct {
        path     string
        cost     int
        selected bool
    }

    vertexes := make([]vertexRecord, g.VertexNum)

    for i, _ := range g.AdjacentList {
        if i == start {
            vertexes[start] = vertexRecord{
                path:     g.getVertex(start).Name,
                cost:     0,
                selected: true,
            }
            continue
        }
        cost, _ := g.getCost(start, i)
        vertexes[i] = vertexRecord{
            path: g.getVertex(start).Name + g.getVertex(i).Name,
            cost: cost,
        }
    }

    lastSelect := start
    for i := 0; i < g.VertexNum-1; i++ {
        minCost := math.MaxInt64
        minCostVertex := 0
        for j, _ := range vertexes {
            if vertexes[j].selected {
                continue
            }
            lastSelectVertex := vertexes[lastSelect]

            cost, isInfinity := g.getCost(lastSelect, j)
            if !isInfinity && lastSelectVertex.cost+cost < vertexes[j].cost {
                vertexes[j].cost = lastSelectVertex.cost + cost
                vertexes[j].path = lastSelectVertex.path + g.getVertex(j).Name
            }

            fmt.Println(fmt.Sprintf(`A到%s最近的路径是%s,其耗费是%d`,
 g.getVertex(j).Name, vertexes[j].path, vertexes[j].cost))
            if vertexes[j].cost < minCost {
                minCost = vertexes[j].cost
                minCostVertex = j
            }
        }
        fmt.Println()
        fmt.Println("本次最短路径为", vertexes[minCostVertex].path, vertexes[minCostVertex].cost)
        fmt.Println()

        vertexes[minCostVertex].selected = true
        lastSelect = minCostVertex
    }

    for i, vertex := range vertexes {
        fmt.Println(i, vertex.path, vertex.cost)
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,530评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,403评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,120评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,770评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,758评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,649评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,021评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,675评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,931评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,751评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,410评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,004评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,969评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,042评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,493评论 2 343