Python图的拓扑排序

Python有向无环图的拓扑排序

拓扑排序的官方定义为:由某个集合上的一个偏序得到该集合上的一个全序,这个操作称之为拓扑排序。而个人认为,拓扑排序即是在图的基本遍历法上引入了入度的概念并围绕入度来实现的排序方法,拓扑排序与Python多继承中mro规则的排序类似,若想深入研究mro规则的C3算法,不妨了解一下 DAG(有向无环图) 的拓扑排序。

入度:指有向图中某节点被指向数目之和
有向无环图:Directed Acyclic Graph,简称DAG,若熟悉机器学习则肯定对DAG不陌生,如ANN、DNN、CNN等则都是典型的DAG模型,对这类模型此处不再过多敷述,有兴趣的可以自行学习。

以一个有向无环图为例,如下图:
DAG(有向无环图)

# 定义图结构
graph = {
    "A": ["B","C"],
    "B": ["D","E"],
    "C": ["D","E"],
    "D": ["F"],
    "E": ["F"],
    "F": [],
}

如图
A的指向的元素为B、C
B的指向的元素为D、E
C的指向的元素为D、E
D的指向的元素为F
E的指向的元素为F
F的指向的元素为空
即A的入度为0,B的入度为1,C的入度为1,D的入度为2,E的入度为2,F的入度为2
在DAG的拓扑排序中,每次都选取入度为 0 的点加入拓扑队列中,再删除与这一点连接的所有边。
首先找到入度为0的点A,把A从队列中取出,同时添加到结果中并把A相关的指向移除,即B、C的入度减少1变为0并将B,C添加到队列中,再从队列首部取出入度为0的节点,以此类推,最后输出结果,完成DAG的拓扑排序。

def TopologicalSort(G):
    # 创建入度字典
    in_degrees = dict((u, 0) for u in G)
    # 获取每个节点的入度
    for u in G:
        for v in G[u]:
            in_degrees[v] += 1
    # 使用列表作为队列并将入度为0的添加到队列中
    Q = [u for u in G if in_degrees[u] == 0]
    res = []
    # 当队列中有元素时执行
    while Q:
        # 从队列首部取出元素
        u = Q.pop(0)
        # 将取出的元素存入结果中
        res.append(u)
        # 移除与取出元素相关的指向,即将所有与取出元素相关的元素的入度减少1
        for v in G[u]:
            in_degrees[v] -= 1
            # 若被移除指向的元素入度为0,则添加到队列中
            if in_degrees[v] == 0:
                Q.append(v)
    return res
print(TopologicalSort(graph))

输出结果:

['A', 'C', 'B', 'E', 'D', 'F']

代码输出结果与上述分析相符

  • 12
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
要使用Python绘制拓扑排序像,可以使用第一段引用中提到的代码作为基础。首先,确保已经定义了一个形结构,并使用该结构来表示要排序的节点和它们之间的关系。 然后,可以使用Matplotlib库来绘制像。以下是一个简单的示例代码,说明如何使用Matplotlib来绘制拓扑排序像: ``` import matplotlib.pyplot as plt def draw_topology_sort(graph): positions = {} # 存储每个节点的位置信息 sorted_nodes = TopologySorted(graph) # 对进行拓扑排序 # 遍历排序后的节点,并确定每个节点的位置 for i, node in enumerate(sorted_nodes): positions[node.value = (i, 0) # 每个节点的位置都在第0行,水平坐标为其在排序后节点列表中的索引 # 绘制节点和边 for node in sorted_nodes: x, y = positions[node.value # 获取节点的坐标 plt.scatter(x, y, marker='o', s=100) # 绘制节点 plt.text(x, y, str(node.value), ha='center', va='bottom') # 在节点上方添加节点值的标签 for next_node in node.nexts: next_x, next_y = positions[next_node.value # 获取下一个节点的坐标 plt.plot([x, next_x], [y, next_y], 'k-') # 绘制边 plt.title('Topology Sort') # 设置像标题 plt.axis('off') # 关闭坐标轴 plt.show() # 显示像 # 使用示例 draw_topology_sort(graph) ``` 这段代码将绘制一个拓扑排序像,其中节点按照它们在排序后的节点列表中的顺序排列,并以该节点的值作为标签显示在节点上方。边将在节点之间绘制。 请注意,该代码仅提供了一个基本的示例,你可以根据自己的需求对其进行修改和扩展,以满足你的具体绘要求。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [(python算法:拓扑排序算法+kruskal算法+prim算法+Dijkstra算法实现](https://blog.csdn.net/qq_31681523/article/details/120181279)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [[Python]的遍历-拓扑排序](https://blog.csdn.net/weixin_30516835/article/details/113508792)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值