码农知识堂 - 1000bd
  •   Python
  •   PHP
  •   JS/TS
  •   JAVA
  •   C/C++
  •   C#
  •   GO
  •   Kotlin
  •   Swift
  • Acwing:食物链(记忆化搜索 Python)


    题目链接🔗:2716. 食物链 - AcWing题库

    解题思路:

    首先我们浏览题目 能得到一个很重要的线索:这是一张拓扑图 即点与点之间存在拓扑关系

    所以我们很自然的能想到使用记忆化搜素,至于原因请参考这篇博客:蓝桥杯/洛谷 : 最长滑雪道/滑雪 + 扩​​​​​​展题目(记忆化搜索 Python)_KS想去海底的博客-CSDN博客d问题描述  小袁非常喜欢滑雪, 因为滑雪很刺激。为了获得速度,滑的区域必须向下倾斜,而且当你滑到坡底,你不得不再次走上坡或者等待升降机来载你。 小袁想知道在某个区域中最长的一个滑坡。区域由一个二维数组给出。数组的每个数字代表点的高度。如下:   一个人可以从某个点滑向上下左右相邻四个点之一,当且仅当高度减小。在上面的例子中,一条可滑行的滑坡为24-17-16-1。当然25-24-23-...-3-2-1更长。事实上,这是最长的一条。  你的任务就是找到最长的一条滑坡,并且将滑坡的长度输出。 滑https://blog.csdn.net/m0_54689021/article/details/125457728?spm=1001.2014.3001.5501大体使用什么算法已经定下来了,接下来应该扣细节了。

    我们可以想一下,怎样才能算作一条食物链,是不是一条从起点能走向终点的路径?

    所以我们可以维护一个dp数组,数组储存从这个点出发(不一定是起点),有几条路径能走到终点。

    那么起点和终点怎么判断呢,这里就需要用到拓扑排序中的术语了:起点(入度为0)

    终点(出度为0)

    确定起点和终点之后 就需要思考这个搜索函数dfs该怎么写了。首先我们可以先思考终点前的一个点,这个点的dp数组值是不是就应该是它连接着几个终点?比如A B都是终点 然后点C->A C->B 有这两条路径,那么dp[C]=2 。

    这样以此类推, 能通向C的点D就加上dp[C]的值 和 从点D能到达的点的dp值,如此往复递归直到起点。

    需要注意的是,如果一个点已经被遍历过了,那么它的dp值就不会再被改变了,因为它的dp值只能被该点的下一个点更改。由于拓扑序的关系,该点之后的点不可能再被遍历到,因此该点的dp值不会再被改变。

    上代码~ 

    1. import sys
    2. sys.setrecursionlimit(1000000) # 修改递归深度 其他语言可以忽略
    3. n,m = map(int,input().split())
    4. Map = [dict() for i in range(n+1)]
    5. din = [0 for i in range(n+1)] # 每个点的入度
    6. dout = [0 for i in range(n+1)] # 每个点的出度
    7. for i in range(m) :
    8. a,b = map(int,input().split())
    9. din[b] += 1 ; dout[a] += 1
    10. Map[a][b] = 1
    11. ST = [] ; EN = [] # 起点 终点
    12. for i in range(1,n+1) :
    13. if din[i] == 0 : ST.append(i)
    14. elif dout[i] == 0 : EN.append(i)
    15. dp = [-1 for i in range(n+1)] # 初始化为-1
    16. def dfs(node) :
    17. global dp
    18. if dp[node] != -1 : # 如果被遍历过直接返回
    19. return dp[node]
    20. if node in EN : return 1 # 如果是终点返回 1
    21. dp[node] = 0
    22. for next in Map[node].keys() : # 否则累加该点能到的所有点的dp值
    23. dp[node] += dfs(next)
    24. return dp[node] # 返回
    25. ans = 0
    26. for i in ST : dfs(i) ; ans += dp[i] # 从每个起点开始走 累加答案
    27. print(ans)

    如有疑问 欢迎留言讨论 ~

  • 相关阅读:
    Vue/Vuex ( modules )核心概念 、 命名空间 namespaced介绍与总结
    vue组件
    谈谈数据分析晓知识
    版本号正则校验及大小比较
    第23期 | GPTSecurity周报
    Java项目:57 ssm011线上旅行信息管理系统ssm+vue
    day 51 |● 503.下一个更大元素II ● 42. 接雨水
    出差学小白知识No5:|Ubuntu上关联GitLab账号并下载项目(ssh key配置)
    springMVC文件上传和下载(简单案例)
    【红外图像】利用红外图像处理技术对不同制冷剂充装的制冷系统进行性能评估(Matlab代码实现)
  • 原文地址:https://blog.csdn.net/m0_54689021/article/details/125485489
  • 最新文章
  • 攻防演习之三天拿下官网站群
    数据安全治理学习——前期安全规划和安全管理体系建设
    企业安全 | 企业内一次钓鱼演练准备过程
    内网渗透测试 | Kerberos协议及其部分攻击手法
    0day的产生 | 不懂代码的"代码审计"
    安装scrcpy-client模块av模块异常,环境问题解决方案
    leetcode hot100【LeetCode 279. 完全平方数】java实现
    OpenWrt下安装Mosquitto
    AnatoMask论文汇总
    【AI日记】24.11.01 LangChain、openai api和github copilot
  • 热门文章
  • 十款代码表白小特效 一个比一个浪漫 赶紧收藏起来吧!!!
    奉劝各位学弟学妹们,该打造你的技术影响力了!
    五年了,我在 CSDN 的两个一百万。
    Java俄罗斯方块,老程序员花了一个周末,连接中学年代!
    面试官都震惊,你这网络基础可以啊!
    你真的会用百度吗?我不信 — 那些不为人知的搜索引擎语法
    心情不好的时候,用 Python 画棵樱花树送给自己吧
    通宵一晚做出来的一款类似CS的第一人称射击游戏Demo!原来做游戏也不是很难,连憨憨学妹都学会了!
    13 万字 C 语言从入门到精通保姆级教程2021 年版
    10行代码集2000张美女图,Python爬虫120例,再上征途
Copyright © 2022 侵权请联系2656653265@qq.com    京ICP备2022015340号-1
正则表达式工具 cron表达式工具 密码生成工具

京公网安备 11010502049817号