news 2026/6/19 23:08:57

0x3f第11天 动态规划课后习题

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
0x3f第11天 动态规划课后习题

1.爬楼梯

1.最关键的一点就是得知道dfs(i)代表的什么

代表一直到台阶i的时候有多少种走法

2.这样就能得到dfs(i)=dfs(i-1)+dfs(i-2)

3.dfs(0)= 1

因为dfs(2)=2 dfs(1)=1 所以dfs(0)必须等于1

回溯代码:

时间复杂度2^n 空间复杂度n

class Solution: def climbStairs(self, n: int) -> int: def dfs(i): if i <=1: return 1 return dfs(i-1)+dfs(i-2) return dfs(n)

记忆型代码:@cache

@cache是装饰器,必须紧贴在被装饰的函数定义上方

时间复杂度n 空间复杂度n

递推型代码:

时间复杂度n 空间复杂度n

class Solution: def climbStairs(self, n: int) -> int: f = [0]*(n+1) f[0]=f[1]=1 for i in range(2,n+1): f[i]=f[i-1]+f[i-2] return f[n]

怎么创建数组f,f=【0】*(n+1),我写的是

f = [0]*n

这样会产生一个长度为n的数组,那就没有f[n]了,最大是f[n-1],所以需要一个长度为n+1的

空间优化:

时间复杂度n 空间复杂度1

class Solution: def climbStairs(self, n: int) -> int: f0=f1=1 for i in range(n-1): newf=f0+f1 f0 = f1 f1 = newf return f1



2.使用最小花费爬楼梯

回溯法:

我错误的地方

if i<=1: return min(cost[0],cost[1])

i为0,或者1 的时候取什么值,读题目没读懂

你可以选择从下标为0或下标为1的台阶开始爬楼梯

结合dfs(i)的定义:到第i层台阶需要花费多少钱

所以dfs(0),dfs(1),到第0,第1层不需要钱

时间2^n,空间n

class Solution: def minCostClimbingStairs(self, cost: List[int]) -> int: n = len(cost) def dfs(i): if i<=1: return min(cost[0],cost[1]) return min(dfs(i-1)+cost[i-1],dfs(i-2)+cost[i-2]) return dfs(n)

记忆性cache:

cache必须依附于装饰的函数

@cache是装饰器,必须紧贴在被装饰的函数定义上方

时间复杂度n 空间复杂度n

class Solution: def minCostClimbingStairs(self, cost: List[int]) -> int: n = len(cost) @cache def dfs(i): if i<=1: return 0 return min(dfs(i-1)+cost[i-1],dfs(i-2)+cost[i-2]) return dfs(n)

递推型代码:

时间复杂度n 空间复杂度n

class Solution: def minCostClimbingStairs(self, cost: List[int]) -> int: n = len(cost) f = [0]*(n+1) f[0]=f[1]=0 for i in range(2,n+1): f[i] = min(f[i-1]+cost[i-1],f[i-2]+cost[i-2]) return f[n]

空间优化:

时间复杂度n 空间复杂度1

class Solution: def minCostClimbingStairs(self, cost: List[int]) -> int: n = len(cost) f0=f1=0 for i in range(2,n+1): newf= min(f1+cost[i-1],f0+cost[i-2]) f0 = f1 f1 = newf return f1



3.爬楼梯Ⅱ

回溯法cache:

多了一个跳三阶,和cost计算公式,其实没啥本质区别,多计算一个dfs(2)

其实也不用

还是只需要计算 i<=0 和i = 1,计算i=2是怕 i-3的时候越界,但其实会归到i<=0的地方

min(dfs(i-1)+costs[i-1]+1,dfs(i-2)+costs[i-1]+4,dfs(i-3)+costs[i-1]+9)

class Solution: def climbStairs(self, n: int, costs: List[int]) -> int: @cache def dfs(i): if i<=0: return 0 if i==1: return costs[0]+1 return min(dfs(i-1)+costs[i-1]+1,dfs(i-2)+costs[i-1]+4,dfs(i-3)+costs[i-1]+9) return dfs(n)

递推:

自己写的代码:ac但是相比灵神冗余,但是对自己来说好理解

class Solution: def climbStairs(self, n: int, costs: List[int]) -> int: f=[0]*(n+1) f[0] = 0 f[1] = costs[0]+1 if n>1: f[2] = min(f[1]+costs[1]+1,f[0]+costs[1]+4) for i in range(3,n+1): f[i] = min(f[i-1]+costs[i-1]+1,f[i-2]+costs[i-1]+4,f[i-3]+costs[i-1]+9) return f[n]

空间优化:太冗余了,看着都想笑

主要就是对i=2的判断

class Solution: def climbStairs(self, n: int, costs: List[int]) -> int: f0 = 0 f1 = costs[0]+1 if n>1: f2 = min(f1+costs[1]+1,f0+costs[1]+4) for i in range(3,n+1): newf= min(f2+costs[i-1]+1,f1+costs[i-1]+4,f0+costs[i-1]+9) f0 = f1 f1 = f2 f2 = newf if n==1: return f1 if n>1: return f2

灵神版:

class Solution: def climbStairs(self, _, costs: List[int]) -> int: f0 = f1 = f2 = 0 for c in costs: f0, f1, f2 = f1, f2, min(f0 + 9, f1 + 4, f2 + 1) + c return f2 。



4.打家劫舍,房子首尾相连

就是在打家劫舍的基础上,多加一个判断nums[0]

如果取了nums[0],做第三座房子到倒数第二座房子的打家劫舍

没取 ,做第二家房子到最后一家房子的打家劫舍


我的错误:因为在rob里面又嵌套了一个rob1,所以要注意参数,里面的就不是nums[ ]

class Solution: def rob(self, nums: List[int]) -> int: n = len(nums) def rob1(arr): f0=f1=0 for x in arr: newf = max(f1,f0+x) f0 = f1 f1 = newf return f1 return max(rob1(nums[2:n-1])+nums[0],rob1(nums[1:n]))



5.mxn棋盘最小路径和(左上走到右下)

回溯法:

1.dfs(i,j)定义:走到grid(i,j)一共走了多少路径

2.边界条件 if i<0 or j <0:

return 多少0?-1?

return inf设置一个 “无效的极大值”,让它在min()比较中被自动忽略

if i==0 and j==0:赋初值

3.怎么得到右下角的坐标,毕竟是mxn的,只知道一个数组

grid = [[1,3,1],[1,5,1],[4,2,1]]

要根据grid求出右下角的坐标

所以 横坐标是 len(grid)-1,纵坐标是len(grid[0])-1

class Solution: def minPathSum(self, grid: List[List[int]]) -> int: @cache def dfs(i,j): if i<0 or j <0: return inf if i == 0 and j == 0: return grid[0][0] return min(dfs(i,j-1),dfs(i-1,j))+grid[i][j] return dfs(len(grid)-1,len(grid[0])-1)

递推:

先根据dfs式子改成f式子

f[i + 1][j + 1] = min(f[i + 1][j], f[i][j + 1]) + grid[i][j]

初始值:f[0][j]=f[i][0]=∞,给整个grid左边上边加一排inf

f[1][1]=grid[0][0]

f[0][1]=f[1][0]=0 ,保证f[1][1] 也可以用递推式计算


难点1.怎么创建一个m*n的f[[][]],并且全部赋初值inf

m是len(grid) n是len(grid[0])

f = [[inf] * (n + 1) for _ in range(m + 1)]

难点2.把每一行的列元素取出来

for i, row in enumerate(grid):
for j, x in enumerate(row):

时间复杂度mn 空间复杂度n

class Solution: def minPathSum(self, grid: List[List[int]]) -> int: m, n = len(grid), len(grid[0]) f = [[inf] * (n + 1) for _ in range(m + 1)] f[0][1] = 0 for i, row in enumerate(grid): for j, x in enumerate(row): f[i + 1][j + 1] = min(f[i + 1][j], f[i][j + 1]) + x return f[m][n]

空间优化:

可以优化到1,看不动了...

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/19 11:20:57

如何正确删除电脑的缓存文件?

新的电脑总是好的&#xff0c;各种干净整洁无垃圾。 还是新的好 表情包 使用了一段时间之后&#xff0c;小伙伴们就会发现电脑C盘飙红了。然后就各种论坛查找清除电脑垃圾的方法。 电脑正常使用下&#xff0c;是会产生很多缓存的&#xff0c;所以C盘红了也很正常。除非电脑组…

作者头像 李华
网站建设 2026/6/19 16:24:55

[python] 代码性能分析工具line_profiler使用指北

码分析能够评估各部分代码的时间消耗&#xff0c;即进行时间复杂度分析。通过这一过程&#xff0c;我们可以识别影响整体运行效率的关键部分&#xff0c;从而更高效地利用底层计算资源。此外&#xff0c;代码分析也可用于评估内存使用情况&#xff0c;即空间复杂度&#xff0c;…

作者头像 李华
网站建设 2026/6/19 12:02:09

《手搓》线程池

一、什么是《手搓》线程池手搓线程池并不是用来完全代替系统线程池的你可以把手搓线程池看做系统线程池的一部分就好比在东海用集装箱搞养殖一个集装箱里养鱼另一个集装箱里养虾搞好隔离,鱼虾都不耽搁二、最常用线程池的场景是什么当然是Task,是用TaskFactory.StartNew方法创建…

作者头像 李华
网站建设 2026/6/19 10:48:44

【MUSIC、最大似然与克拉美-罗下界】MUSIC与ESPRIT 算法来估计到达角(AoA),并尝试推导克拉美-罗下界(CRLB)以分析其性能研究附Matlab代码

✅作者简介&#xff1a;热爱科研的Matlab仿真开发者&#xff0c;擅长数据处理、建模仿真、程序设计、完整代码获取、论文复现及科研仿真。&#x1f34e; 往期回顾关注个人主页&#xff1a;Matlab科研工作室&#x1f34a;个人信条&#xff1a;格物致知,完整Matlab代码及仿真咨询…

作者头像 李华
网站建设 2026/6/19 14:24:10

LLM 场景下的强化学习技术扫盲

强化学习基础&#xff1a;行业黑话想象你正在和一个刚训练好的语言模型聊天。你问&#xff1a;“今天过得怎么样&#xff1f;”模型可能回&#xff1a;“还行。” 也可能回&#xff1a;“我是个 AI&#xff0c;没有感情。”人类觉得前者更自然、更友好——这就是偏好反馈。强化…

作者头像 李华