news 2026/4/17 20:30:31

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/9 8:14:01

医疗AI智能体架构设计:六大核心模块与七种专业智能体类型全解析

文章介绍了医疗AI智能体的六大核心模块框架&#xff1a;感知、对话接口、交互系统、工具集成、记忆学习和推理&#xff0c;以及七种专业智能体类型的特点与应用场景。这一模块化架构旨在构建安全、可解释且自适应的医疗AI系统&#xff0c;推动人工智能在医疗领域的深度应用&…

作者头像 李华
网站建设 2026/4/17 6:38:33

js函数声明和函数表达式的理解

在JS中,函数声明会被提升&#xff0c;这意味着函数可以在声明之前被调用。当你使用函数声明的方式定义函数: function resizeFn() {...}整个函数声明会被提升到作用域的顶部。这意味着在整个作用域内&#xff0c;无论函数在何处声明&#xff0c;都可以在声明前调用。函数声明会…

作者头像 李华
网站建设 2026/4/18 3:27:40

奶奶都能看懂的 C++ —— vector 与迭代器

但是在讲解它之前&#xff0c;我们需要先了解迭代的对象是什么。常见的一种&#xff0c;叫做 vector。vector 类型使用可变有序序列我们知道&#xff0c;数学里&#xff0c;vector 是向量的意思。但 C 里的向量和它不太一样。它的含义是&#xff0c;具有可变元素个数的有序对象…

作者头像 李华
网站建设 2026/4/18 3:37:33

基于java的SpringBoot/SSM+Vue+uniapp的校园活动管理系统的详细设计和实现(源码+lw+部署文档+讲解等)

文章目录前言详细视频演示具体实现截图技术栈后端框架SpringBoot前端框架Vue持久层框架MyBaitsPlus系统测试系统测试目的系统功能测试系统测试结论为什么选择我代码参考数据库参考源码获取前言 &#x1f31e;博主介绍&#xff1a;✌全网粉丝15W,CSDN特邀作者、211毕业、高级全…

作者头像 李华
网站建设 2026/4/17 19:36:30

MPK(Mirage Persistent Kernel)源码笔记(1)--- 基础原理

x00 概要 CMU 贾志豪老师团队提出的MPK&#xff08;Mirage Persistent Kernel&#xff09;是依托 Mirage 编译器生态的创新运行时系统&#xff0c;其核心能力在于将多GPU环境下大语言模型&#xff08;LLM&#xff09;推理任务自动转换为适配GPU架构的高性能巨型内核&#xff0…

作者头像 李华