news 2026/4/15 22:10:17

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/4/15 16:15:06

Spring Boot快速集成MiniMax、CosyVoice实现文本转语音

在一些需要高质量文本转语音&#xff08;TTS&#xff09;的场景中&#xff08;比如&#xff1a;有声书配音、播客等&#xff09;。之前介绍的EdgeTTS方案可能效果没有那么好。此时就比较推荐使用 MiniMax、CosyVoice这些提供的音色&#xff0c;这些音色的效果会更加拟人、逼真&…

作者头像 李华
网站建设 2026/4/14 13:19:22

逆向提示法:让大模型输出从平庸到专业的5步技巧

文章介绍"逆向提示"技巧&#xff0c;通过提供满意样例让模型反推提示词配方&#xff0c;解决AI内容同质化问题。该方法提炼语气、节奏、结构等要素&#xff0c;形成可复用模板&#xff0c;显著提升内容质量与一致性。作者提供社媒文案、产品描述等多场景应用案例&…

作者头像 李华
网站建设 2026/4/10 14:49:07

算法分析--基数排序

时间复杂度 O&#xff08;KN&#xff09;线性高位优先&#xff08;不好&#xff09;先按照高位升序排序&#xff0c;依次进行下去&#xff0c;直到排到最低位。image因为高位有一个分组的动作&#xff0c;在每个组里面对低位再排序。可以用递归。实际上&#xff0c;完全可以用低…

作者头像 李华
网站建设 2026/4/15 15:04:43

UVa 10641 Barisal Stadium

题目描述 孟加拉板球控制委员会决定在巴里萨尔建造一座新的国际板球场。该体育场形状为凸多边形&#xff0c;需要在外部安装泛光灯以便在灯光下比赛。每个泛光灯可以照亮体育场的某些边&#xff0c;建造每个灯需要一定成本。 照亮条件 &#xff1a;一条边被某个灯照亮&#xff…

作者头像 李华
网站建设 2026/4/13 16:53:01

AgentScope深入分析-设计模式与架构决策分分析

设计的精髓&#xff1a;设计模式与架构决策分析 摘要 AgentScope 的设计体现了深厚的工程智慧。本文将深入分析框架中使用的设计模式、架构决策&#xff0c;以及这些设计背后的考量。你会发现&#xff0c;框架大量使用了模板方法模式、策略模式、观察者模式、元类模式等经典设计…

作者头像 李华
网站建设 2026/4/15 4:04:56

MySQL的这6大雷区,大部分人都会踩中!

苏三的工作内推群为什么MySQL雷区如此之多&#xff1f;在深入具体雷区之前&#xff0c;我们先聊聊为什么MySQL这么容易踩坑。这背后有几个深层次原因&#xff1a;看似简单&#xff1a;MySQL语法简单&#xff0c;入门容易&#xff0c;让很多人低估了它的复杂性默认配置坑多&…

作者头像 李华