拓扑排序的实现(python)

1 AOV网(Activity On Vertex Network)

一项大的工程常被分为多个小的子工程,子工程之间可能存在一定的先后顺序,即某些子工程必须在其他的一些子工程完成后才能开始。
在现代化管理中,人们常用有向图来描述和分析一项工程的计划和实施过程,子工程被称为活动(Activity)。以顶点表示活动、有向边表示活动之间的先后关系,这样的图简称为 AOV 网。
标准的AOV网必须是一个有向无环图(Directed Acyclic Graph,简称 DAG),如下图所示:
在这里插入图片描述

在该图中,B依赖于A的完成,C依赖于A的完成,D依赖于B的完成,E依赖于A的完成,F依赖于C、D、E的完成。

2 拓扑排序(Topological Sort)

拓扑排序是将 AOV 网中所有活动排成一个序列,使得每个活动的前驱活动都排在该活动的前面。比如上图的拓扑排序结果是:A、B、C、D、E、F 或者 A、B、D、C、E、F (结果并不一定是唯一的)。

2.1 实现思路

可以使用卡恩算法(Kahn于1962年提出)完成拓扑排序。卡恩算法的思路为:首先访问入度为0的顶点,也就是没有任何前驱依赖的节点,访问结束后删除这些节点和关联的边,并更新这些边末端到达顶点的入度;之后,再寻找未访问但入度为0的顶点,进行重复操作。示例如下:
在这里插入图片描述

我们先寻找入度为0的点,此时为A,访问其元素并删除该点与相关的边,相应的,更新这些表终点的入度。

在这里插入图片描述

此时存在多个入度为0的顶点,我们可以任意选一个,删除节点与边,在此选择C。

在这里插入图片描述

再选择B。

在这里插入图片描述

再选择E。

在这里插入图片描述

再选择D。

在这里插入图片描述

最后就只剩下F了。

在这里插入图片描述
还需要注意的是,如果此时 输出的元素个数少于顶点总数,说明原图中存在环,无法进行拓扑排序。

2.2 代码实现

由于卡恩算法会将顶点和边都进行删除,而这将导致图结构发生变化,因此我们在代码实现时并非真的将顶点和边删除,而是对所记录的每个顶点的入度进行修改,当然,在一开始我们需要一张表格记录每个顶点的入度,这点我们可以通过哈希表map来实现。

在这里插入图片描述
此外,可能同时同时存在多个入度为0的顶点,为此我们需要一个队列来逐个处理。而在处理完当前顶点后,我们还需要对其子顶点的入度进行更新,若入度为0则进行下一步访问,而这显然是一个广度优先搜索的过程。因此,使用bfs就可以轻松完成拓扑排序了,具体代码见下。关于该图的具体定义见:使用邻接表实现数据结构图(python)

class Graph():

	...
	
    def topological_sort(self):
        print('topological sort----------------')
        in_degree_map={}#入度map
        queue=Queue()
        result=[]
        for vertex in self.vertices.values():#构造入度矩阵
            if len(vertex.inEdges)==0:
                queue.put(vertex)#入队
            else:
                in_degree_map[vertex]=len(vertex.inEdges)

        while not queue.empty():
            vertex=queue.get()
            result.append(vertex.value)
            for edge in vertex.outEdges:
                in_degree =in_degree_map.get(edge.to)-1
                in_degree_map[edge.to]=in_degree
                if in_degree==0:
                    queue.put(edge.to)

        if len(result)<len(self.vertices):
            print('图有环,无法进行拓扑排序')
        else:
            print('图无环,拓扑排序结果为:')
            print(result)
  • 8
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
拓扑排序是一种对有向无环图(DAG)进行排序的算法,它可以确定图中节点的线性顺序,使得对于图中的每一条有向边 (u, v),节点 u 在排序中都排在节点 v 的前面。在Python中,我们可以使用广度优先搜索(BFS)或深度优先搜索(DFS)来实现拓扑排序。 一种方法是使用广度优先搜索(BFS)。我们首先构建一个入度矩阵,记录每个节点的入度(即指向该节点的边的数量)。然后,我们创建一个队列,并将入度为0的节点加入队列中。接下来,我们循环处理队列中的节点,将其添加到结果列表中,并更新其子节点的入度。如果子节点的入度变为0,则将其加入队列。如果最终结果列表的长度小于图中节点的数量,则说明图中存在环。否则,结果列表就是拓扑排序的结果。 下面是使用广度优先搜索实现拓扑排序Python代码示例: ``` # 构建图类 class Graph(): ... def topological_sort(self): in_degree_map = {} # 入度map queue = Queue() result = [] # 构造入度矩阵 for vertex in self.vertices.values(): if len(vertex.inEdges) == 0: queue.put(vertex) # 入队 else: in_degree_map[vertex = len(vertex.inEdges) while not queue.empty(): vertex = queue.get() result.append(vertex.value) for edge in vertex.outEdges: in_degree = in_degree_map.get(edge.to) - 1 in_degree_map[edge.to = in_degree if in_degree == 0: queue.put(edge.to) if len(result) < len(self.vertices): print('图有环,无法进行拓扑排序') else: print('图无环,拓扑排序结果为:') print(result) ``` 另一种方法是使用深度优先搜索(DFS)。我们可以给每个节点标记3种状态:未访问、正在访问和已访问。在DFS过程中,我们首先将节点标记为正在访问,然后递归地访问其邻居节点。当我们遇到一个已经标记为正在访问的节点时,说明图中存在环,无法进行拓扑排序。当我们访问完一个节点的所有邻居节点后,我们将该节点标记为已访问,并将其添加到结果列表的开头。最后,结果列表的顺序就是拓扑排序的结果。 这是使用深度优先搜索实现拓扑排序Python代码示例: ``` # 构建图类 class Graph(): ... def topological_sort_dfs(self): color_map = {} # 节点颜色map result = [] def dfs(vertex): nonlocal result color_map[vertex = 'gray' # 标记为正在访问 for edge in vertex.outEdges: if color_map.get(edge.to) == 'gray': print('图有环,无法进行拓扑排序') return elif color_map.get(edge.to) == 'white': dfs(edge.to) color_map[vertex = 'black' # 标记为已访问 result.insert(0, vertex.value) for vertex in self.vertices.values(): color_map[vertex = 'white' # 标记为未访问 for vertex in self.vertices.values(): if color_map[vertex == 'white': dfs(vertex) print('图无环,拓扑排序结果为:') print(result) ``` 以上是两种使用Python实现拓扑排序的方法:广度优先搜索(BFS)和深度优先搜索(DFS)。具体选择哪种方法取决于实际需求和图的特点。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值