python实现生成树

生成树

生成树(Spanning Tree)是一个连通图的生成树是图的极小连通子图,它包含图中的所有顶点,并且只含尽可能少的边。这意味着对于生成树来说,若砍去它的一条边,则会使生成树变成非连通图;若给它增加一条边,则会形成图中的一条回路。

最小生成树

最小生成树(Minimum Spanning Tree,简称 MST)是一个图的生成树中,边的权重之和最小的那棵生成树。

对于一个带权连通无向图G=(V,E),生成树不同,每棵树的权(即树中所有边上的权值之和)也可能不同。设X为G的所有生成树的集合,若T为X中边的权值之和最小的那棵生成树,则T称为G的最小生成树(Minimum-Spanning-Tree(MST),在一个加权连通图中,可能存在多个不同的生成树,但是其中只有一个最小生成树。最小生成树通常用于解决网络设计、通信网络等问题。

不难看出,最小生成树具有如下性质:

1)最小生成树不是唯一的,即最小生成树的树形不唯一,X中可能有多个最小生成树。当图G中的各边权值互不相等时,G的最小生成树是唯一的;若无向连通图G的边数比顶点数少1,即G本身是一棵树时,则G的最小生成树就是它本身。

2)最小生成树的边的权值之和总是唯一的,虽然最小生成树不唯一,但其对应的边的权值之和总是唯一的,而且是最小的。

3)最小生成树的边数为顶点数减1

构造最小生成树有多种算法,但大多数算法都利用了最小生成树的下列性质:假设G=(V,E)是一个带权连通无向图,U是顶点集V的一个非空子集。若(u,v)是一条具有最小叔值的边,其中u∈U, v∈V- U,则必存在一棵包含边(u,V)的最小生成树,基于该性质的最小生成树算法主要有Prim算法和Kruskal算法,它们都基于贪心算法的策略。

常用的最小生成树算法

Prim算法:Prim算法是一种贪心算法,从一个顶点开始,每次选择权重最小的边来扩展最小生成树,直到所有顶点都加入到最小生成树中为止。

Kruskal算法:Kruskal算法是一种基于并查集的贪心算法,它首先将所有边按权重从小到大排序,然后依次考虑每条边,如果当前边连接的两个顶点不在同一个连通分量中,则将这条边加入最小生成树中,并将这两个顶点合并到同一个连通分量中,直到最小生成树的边数达到n-1为止。

最小生成树算法的选择:

如果图的边数量比较少,那么Kruskal算法通常更加简洁高效。

如果图的顶点数量比较少,那么Prim算法可能更容易实现和理解。

如果图是稠密图(边数量接近于完全图),那么Prim算法的时间复杂度可能更低,因为Prim算法在每一步都只需要考虑与当前最小生成树相邻的边。

复制代码
from heapq import heappop, heappush

# Prim算法
def prim(graph):
    n = len(graph)
    visited = [False] * n
    min_heap = [(0, 0)]  # (权重, 顶点)
    mst_weight = 0
    while min_heap:
        weight, node = heappop(min_heap)
        if visited[node]:
            continue
        visited[node] = True
        mst_weight += weight
        for neighbor, weight in graph[node]:
            if not visited[neighbor]:
                heappush(min_heap, (weight, neighbor))
    return mst_weight

# Kruskal算法
def kruskal(graph):
    n = len(graph)
    parent = list(range(n))
    edges = []
    mst_weight = 0
    for u in range(n):
        for v, weight in graph[u]:
            edges.append((weight, u, v))
    edges.sort()
    for weight, u, v in edges:
        if find(u, parent) != find(v, parent):
            union(u, v, parent)
            mst_weight += weight
    return mst_weight

def find(x, parent):
    if parent[x] != x:
        parent[x] = find(parent[x], parent)
    return parent[x]

def union(x, y, parent):
    root_x = find(x, parent)
    root_y = find(y, parent)
    parent[root_x] = root_y

# 测试
graph = [
    [(1, 1), (2, 2)],
    [(0, 1), (2, 4), (3, 5)],
    [(0, 2), (1, 4), (3, 3)],
    [(1, 5), (2, 3)]
]

print("Prim算法最小生成树权重:", prim(graph))
print("Kruskal算法最小生成树权重:", kruskal(graph))

参考链接:https://zhuanlan.zhihu.com/p/136387766

相关推荐
大白同学42115 分钟前
【C++】C++11介绍(Ⅱ)
开发语言·c++
你怎么知道我是队长23 分钟前
C语言---存储类
c语言·开发语言
XIAOYU67201329 分钟前
金融数学专业需要学哪些数学和编程内容?
开发语言·matlab·金融
油炸自行车32 分钟前
【Qt】编写Qt自定义Ui控件步骤
开发语言·c++·qt·ui·自定义ui控件·qt4 自定义ui控件
夏日麋鹿~35 分钟前
逐时nc数据批量处理为日平均
python
程序员三明治40 分钟前
Python编辑器的安装及配置(Pycharm、Jupyter的安装)从0带你配置,小土堆视频
python·pycharm·编辑器
理想国的女研究僧41 分钟前
Jupyter Notebook操作指南(1)
ide·python·学习·jupyter
酷飞飞1 小时前
PyQt 界面布局与交互组件使用指南
python·qt·交互·pyqt
浪扼飞舟1 小时前
c#基础二(类和对象,构造器调用顺序、访问级别、重写和多态、抽象类和接口)
java·开发语言·c#
GilgameshJSS1 小时前
【学习K230-例程19】GT6700-TCP-Client
网络·python·网络协议·学习·tcp/ip