Python 蓝桥杯之拓扑排序 检测环

简介: Python 蓝桥杯之拓扑排序 检测环

了解拓扑排序之前,先了解AOV网:

image.png

有向图中若以顶点表示活动,有向边表示活动之间的先后关系,这样的图简称为AOV网

比如C1指向C4,说明C4活动的开展以C1的开展为前提(C1没完成就不能去弄C4)

知道了AOV网图,就可以引入拓扑序列的概念:

拓扑序列:拓扑序列是顶点活动网(AOV网)中将活动按发生的先后次序进行的一种排列

这么说有点抽象,传送门:某站上视频对这个讲解的很好

值得一提的是:任何一个有向无环图一定有拓扑序列,因此拓扑序列可以用来检测环

拓扑排序之前,先了解有向图的三个基本概念:入度,出度,度

入度:指向该顶点的入边 出度:从该顶点指出的边    度:入度和出度的总和(与顶点相关联的边的条数)

从图上来看,拓扑排序需要这么做(视频里也有详细的分析过程,建议去看,强烈建议):

image.png

那么对于代码来说,我们需要运用深搜的思想来执行拓扑排序的操作,1:找到图中入度为0的顶点 压栈 2:处于栈顶的顶点出栈 并加入列表(另外创建的存储结构,用于存储拓扑序列中的元素) 访问该顶点的所有邻居,如果其存在入度为0的邻居,压栈,执行操作2,如果不存在,回溯。


下面以上图为例,将C1,C2...C7依次对应为字母abcdefg,给出创建拓扑序列的代码:

#准备工作上需要一个字典:用于存放连接关系
def topsort(graph):
    #初始化所有点的入度为0
    indegrees=dict((i,0) for i in graph.keys())
    #传入入度大小
    for i in graph.keys():
        for j in graph[i]:
            indegrees[j]+=1#'a':'cd',代表a指向c和d,那么c和d的入度增加1
    #获取入度为0的顶点
    stack=[i for i in indegrees.keys() if indegrees[i]==0]
    #创建拓扑序列
    seq=[]
    while stack:#深搜
        tmp=stack.pop()#出栈
        seq.append(tmp)#加入拓扑序列
        for k in graph[tmp]:#遍历邻居,邻居入度-1,邻居入度为0入栈
            indegrees[k]-=1
            if indegrees[k]==0:
                stack.append(k)
    if len(seq)==len(graph):
        print(seq)#输出拓扑序列
    else:
        print('存在环')#当存在环,最终余下的点入度都不为0,Seq个数少于总顶点数
G={
    'a':'cd',
    'b':'df',
    'c':'e',
    'd':'efg',
    'e':'g',
    'f':'g',
    'g':''
}
topsort(G)
#['b', 'a', 'd', 'f', 'c', 'e', 'g']


蓝桥杯真题演练:


image.png


第一次没优化只有71分,优化了第二次是85。(这最后一个如有大佬知道怎么过的请教一下)

image.png

代码设计思路:找能够组成圈的所有点,根据这些点找到最大环。


首先解决如何计算圈的大小问题:比如A结点经过一系列结点最后回到了A结点,我们要知道圈的大小,只需要知道这一过程中A走过了几个不同的结点,换句话说,这个圈内有几个元素(集合的思想)。然后比较各个圈的大小即可知道最大环。


这里有一点需要强调:虽然我们可以通过遍历每个点(能够组成圈的点),枚举出所有圈大小的情况,但是会有重复,当数据大起来,就会超时(71)。那怎么样才能避免重复呢,思考,比如题目给出的实例,2,3,4,5组成一个圈,假设我在遍历结点2的时候,计算出了这个圈的大小是4,那么圈内的结点,在下一次访问都不需要再访问了,也就是说下一次可以直接访问结点6。


原因:问题就在于圈内的结点有没有可能与其他结点组成更大的圈,如果我们不再访问,会不会导致范围缩小或者遗漏?答案是不会的。因为,如果要组成更大的圈,圈内的结点必然要与圈外的结点连接。由于圈内的结点是作为圈的一部分,那么必定已经有一条边指向这个结点。如果存在这个结点与圈外的结点构成更大的圈,那么必然与圈外的结点有连接,那么这个结点就会同时指向两个结点,就不符合题意了。


#准备工作上需要一个字典:用于存放连接关系
def topsort(graph):
    #初始化所有点的入度为0
    indegrees=dict((i,0) for i in graph.keys())
    #传入入度大小
    for i in graph.keys():
        indegrees[graph[i]]+=1
    #获取入度为0的顶点
    stack=[i for i in indegrees.keys() if indegrees[i]==0]
    #创建拓扑序列
    seq=[]
    while stack:#深搜
        tmp=stack.pop()#出栈
        seq.append(tmp)#加入拓扑序列
        for k in graph[tmp]:#遍历邻居,邻居入度-1,邻居入度为0入栈
            indegrees[k]-=1
            if indegrees[k]==0:
                stack.append(k)
    if len(seq)==len(graph):
        print(seq)#对于本题而言一定有圈
    else:
        ans=0#最大圈的大小
        res=[i for i in indegrees if indegrees[i]!=0]#环上的点
        found=[]#用于减少次数:已查找过的点
        for j in res:#寻找最大环
            if j not in found:
                start=j
                p=set()
                while G[j]!=start:
                    found.append(j)
                    p.update([int(j)])
                    j=G[j]
                found.append(j)
                p.update([int(j)])
                ans=max(ans,len(p))
        print(ans)
n=int(input().strip())
m=list(map(int,input().strip().split()))
G={}
for i in enumerate(m):
    G[str(i[0]+1)]=str(i[1])
topsort(G)


我是小郑 正在奔赴热爱奔赴山海!


目录
相关文章
|
25天前
|
机器学习/深度学习 监控 TensorFlow
使用Python实现深度学习模型:智能农业病虫害检测与防治
使用Python实现深度学习模型:智能农业病虫害检测与防治
180 65
|
13天前
|
机器学习/深度学习 数据采集 算法
时间序列结构变化分析:Python实现时间序列变化点检测
在时间序列分析和预测中,准确检测结构变化至关重要。新出现的分布模式往往会导致历史数据失去代表性,进而影响基于这些数据训练的模型的有效性。
30 1
|
12天前
|
Python
Python中几种lambda排序方法
【9月更文挑战第7天】在Python中,`lambda`表达式常用于配合排序函数,实现灵活的数据排序。对于基本列表,可以直接使用`sorted()`进行升序或降序排序;处理复杂对象如字典列表时,通过`lambda`指定键值进行排序;同样地,`lambda`也适用于根据元组的不同位置元素来进行排序。
|
22天前
|
数据采集 人工智能 数据可视化
Python selenium爬虫被检测到,该怎么破?
Python selenium爬虫被检测到,该怎么破?
|
24天前
|
算法 数据可视化 安全
使用Python检测新冠肺炎疫情拐点,抗疫成果明显
使用Python检测新冠肺炎疫情拐点,抗疫成果明显
|
20天前
|
Python
Python魔法:用一行代码实现数据排序
【8月更文挑战第31天】忘掉传统多行排序代码,本文揭秘如何使用一行Python代码快速对数据进行排序,同时深入探讨背后的原理和性能考量。
|
1月前
|
机器学习/深度学习 算法 大数据
【2023年MathorCup高校数学建模挑战赛-大数据竞赛】赛道A:基于计算机视觉的坑洼道路检测和识别 python 代码解析
本文提供了2023年MathorCup高校数学建模挑战赛大数据竞赛赛道A的解决方案,涉及基于计算机视觉的坑洼道路检测和识别任务,包括数据预处理、特征提取、模型建立、训练与评估等步骤的Python代码解析。
44 0
【2023年MathorCup高校数学建模挑战赛-大数据竞赛】赛道A:基于计算机视觉的坑洼道路检测和识别 python 代码解析
|
1月前
|
Python
【Python】对字典进行排序
该文档介绍了如何在Python中对字典进行排序的方法。
16 2
|
2月前
|
机器学习/深度学习 运维 监控
使用Python实现深度学习模型:智能安防监控与异常检测
【7月更文挑战第26天】 使用Python实现深度学习模型:智能安防监控与异常检测
49 6
|
2月前
|
Python
Python小技巧:一种字符串的排序方式
Python小技巧:一种字符串的排序方式
19 0