算法训练营day34 动态规划② 62.不同路径、63. 不同路径 II、343整数拆分、96.不同的二叉搜索树

        动态规划的第二篇博客!进阶题目,有一说一,尤其最后一道题,真的难想到这种解法

        找规律!!!

62.不同路径

        注意本题是路径不是路程!!!

动态规划

  • 确定dp数组(dp table)以及下标的含义

        dp[i][j] :表示从(0 ,0)出发,到(i, j) 有dp[i][j]条不同的路径。

  • 确定递推公式

        想要求dp[i][j],只能有两个方向来推导出来,即dp[i - 1][j] 和 dp[i][j - 1]。

        那么很自然,dp[i][j] = dp[i - 1][j] + dp[i][j - 1],因为dp[i][j]只有这两个方向过来。

  • dp数组的初始化

        dp[i][0]一定都是1,因为从(0, 0)的位置到(i, 0)的路径只有一条,那么dp[0][j]也同理

  • 确定遍历顺序

        这里要看一下递推公式dp[i][j] = dp[i - 1][j] + dp[i][j - 1],dp[i][j]都是从其上方和左方推导而来,那么从左到右一层一层遍历就可以了。

        这样就可以保证推导dp[i][j]的时候,dp[i - 1][j] 和 dp[i][j - 1]一定是有数值的。

  • 举例推导dp数组(后续略)

class Solution:def uniquePaths(self, m: int, n: int) -> int:dp = [[0] * n for _ in range(m)]for i in range(m):dp[i][0] = 1for j in range(n):dp[0][j] = 1for i in range(1, m):for j in range(1, n):dp[i][j] = dp[i - 1][j] + dp[i][j - 1] # 核心逻辑, 路径数为上节点数 + 左节点数return dp[m - 1][n - 1]'''
# 空间优化
# 仅留最后一行数据, 其他行数据用于过程计算
class Solution:def uniquePaths(self, m: int, n: int) -> int:# 创建一个一维列表用于存储每列的唯一路径数dp = [1] * n# 计算每个单元格的唯一路径数for j in range(1, m):for i in range(1, n):dp[i] += dp[i - 1]# 返回右下角单元格的唯一路径数return dp[n - 1]
'''

数论

        在这个图中,可以看出一共m,n的话,无论怎么走,走到终点都需要 m + n - 2 步。在这m + n - 2 步中,一定有 m - 1 步是要向下走的,不用管什么时候向下走。

        那么有几种走法呢? 可以转化为,给你m + n - 2个不同的数,随便取m - 1个数,有几种取法。那么这就是一个组合问题了。

class Solution:def uniquePaths(self, m: int, n: int) -> int:# 核心是一个组合算法的实现numerator = 1  # 分子denominator = m - 1  # 分母count = m - 1  # 计数器,表示剩余需要计算的乘积项个数t = m + n - 2  # 初始乘积项while count > 0:numerator *= t  # 计算乘积项的分子部分t -= 1  # 递减乘积项while denominator != 0 and numerator % denominator == 0:numerator //= denominator  # 约简分子denominator -= 1  # 递减分母count -= 1  # 计数器减1,继续下一项的计算return numerator  # 返回最终的唯一路径数

深度搜索

        题目中说机器人每次只能向下或者向右移动一步,那么其实机器人走过的路径可以抽象为一棵二叉树,而叶子节点就是终点!

        最后来看这个代码就很简单了

class Solution:def uniquePaths(self, m: int, n: int) -> int:if m == 1 or n == 1:return 1return self.uniquePaths(m - 1, n) + self.uniquePaths(m, n - 1)

63. 不同路径 II

        相对上题增加了障碍,有障碍的话,其实就是标记对应的dp table(dp数组)保持初始值(0)就可以了。

  • 确定dp数组(dp table)以及下标的含义

        dp[i][j] :表示从(0 ,0)出发,到(i, j) 有dp[i][j]条不同的路径。

  • 确定递推公式

        递推公式和上题一样,dp[i][j] = dp[i - 1][j] + dp[i][j - 1]。

        但这里需要注意一点,因为有了障碍,(i, j)如果就是障碍的话应该就保持初始状态(初始状态为0)。

  • dp数组如何初始化

        因为从(0, 0)的位置到(i, 0)的路径只有一条,所以dp[i][0]一定为1,dp[0][j]也同理。

        但如果(i, 0) 这条边有了障碍之后,障碍之后(包括障碍)都是走不到的位置了,所以障碍之后的dp[i][0]应该还是初始值0。

  • 确定遍历顺序

        从递归公式dp[i][j] = dp[i - 1][j] + dp[i][j - 1] 中可以看出,一定是从左到右一层一层遍历,这样保证推导dp[i][j]的时候,dp[i - 1][j] 和 dp[i][j - 1]一定是有数值。

class Solution:def uniquePathsWithObstacles(self, obstacleGrid: List[List[int]]) -> int:m = len(obstacleGrid)n = len(obstacleGrid[0])if obstacleGrid[m - 1][n - 1] == 1 or obstacleGrid[0][0] == 1:return 0# 终点和起点有阻碍dp = [[0] * n for _ in range(m)]for i in range(m):if obstacleGrid[i][0] == 0:dp[i][0] = 1else:breakfor j in range(n):if obstacleGrid[0][j] == 0:dp[0][j] = 1else:breakfor i in range(1, m):for j in range(1, n):if obstacleGrid[i][j] == 1:continuedp[i][j] = dp[i - 1][j] + dp[i][j - 1]return dp[m - 1][n - 1]

343整数拆分

  • 确定dp数组(dp table)以及下标的含义

        dp[i]:分拆数字i,可以得到的最大乘积为dp[i]。

  • 确定递推公式

        从1遍历j,有两种渠道得到dp[i].

  1. 一个是j * (i - j) 直接相乘。(拆分为两个数)
  2. 一个是j * dp[i - j],相当于是拆分(i - j) (拆分为3个及3个以上的数)

        所以递推公式:dp[i] = max({dp[i], (i - j) * j, dp[i - j] * j});在递推公式推导的过程中,增加参数dp[i],取最大。

  • dp的初始化

        dp[0] dp[1] 不应该初始化,只初始化dp[2] = 1

  • 确定遍历顺序

        dp[i] 是依靠 dp[i - j]的状态,所以遍历i一定是从前向后遍历,先有dp[i - j]再有dp[i]。

class Solution:def integerBreak(self, n: int) -> int:# 假设对正整数 i 拆分出的第一个正整数是 j(1 <= j < i),则有以下两种方案:# 1) 将 i 拆分成 j 和 i−j 的和,且 i−j 不再拆分成多个正整数,此时的乘积是 j * (i-j)# 2) 将 i 拆分成 j 和 i−j 的和,且 i−j 继续拆分成多个正整数,此时的乘积是 j * dp[i-j]dp = [0] * (n + 1)dp[2] = 1for i in range(3, n + 1):# 开始计算具体的dp[i]for j in range(1, i // 2 + 1):dp[i] = max(dp[i], (i - j) * j, dp[i - j] * j)return dp[n]

96.不同的二叉搜索树 

很明显的动态规划题目,但是递推公式确实很难想到

  • 确定dp数组(dp table)以及下标的含义

        dp[i] : 1到i为节点组成的二叉搜索树的个数为dp[i]。

  • 确定递推公式

        在上面的分析中,其实已经看出其递推关系, dp[i] += dp[以j为头结点左子树节点数量] * dp[以j为头结点右子树节点数量]

        j相当于是头结点的元素,从1遍历到i为止。eg3: 2.0 + 1.1 + 0.2 

  • dp数组如何初始化

        初始化dp[0] = 1

  • 确定遍历顺序

        节点数为i的状态是依靠 i之前节点数的状态。所以我认为仍然可以理解为自前向后遍历

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/bicheng/91102.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/91102.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring 5 事务详解

一、核心使用方式声明式事务&#xff08;推荐&#xff09;通过 Transactional 注解实现&#xff0c;需配合配置启用&#xff1a;Configuration EnableTransactionManagement public class AppConfig {Beanpublic PlatformTransactionManager txManager(DataSource dataSource) …

[ctfshow web入门]web99 in_array的弱比较漏洞

信息收集 array_push(array, value)&#xff1a;向数组最后的位置插入value in_array(value, array, type)&#xff1a;其中value是要查找的值&#xff0c;array是需要查找的的数组&#xff0c;type是查找的类型&#xff0c;如果没有指定类型&#xff0c;则以弱比较方式查找 i…

mysql5.6 常用查询sql

mysql5.6 常用查询sql 文章目录 mysql5.6 常用查询sql 1.查询版本 2.MySQL 运行状态(Ping) 3.慢查询数量 4.连接数 5.最大连接数 6.InnoDB 缓冲池命中率 7.表锁等待次数 8.二进制日志状态 9.表空间使用率 10.查询缓存效率 11.每次自动扩展空间大小 12.导入导出 ✅ 一、导出(…

【在Unity游戏开发中Dictionary、List介绍】

在Unity游戏开发中&#xff0c;Dictionary和List是最核心的两种数据结构&#xff0c;它们各自有优势和应用场景。以下是介绍&#xff1a;&#x1f9e0; 数据结构本质对比特性Dictionary<TKey, TValue>List底层结构哈希表&#xff08;Hash Table&#xff09;动态数组&…

windows平台计划任务批处理实现定时任务

无限循环加定时延时计划任务用户登录执行一次下文中300代表300秒执行一次第2,3,4行为vbs隐藏窗口C:\me\corn\test.batecho off if "%1""hide" goto CmdBegin start mshta vbscript:createobject("wscript.shell").run("""%~0&quo…

深入理解 TCP 协议:从原理到实践的技术解析

目录 一、TCP 协议的核心定位与特性 1.1 协议栈中的位置 1.2 五大核心特性 二、TCP 连接建立与终止的底层逻辑 2.1 三次握手&#xff08;连接建立&#xff09; 2.2 四次挥手&#xff08;连接终止&#xff09; 三、TCP 可靠传输的核心机制 3.1 序列号与确认机制 3.2 滑…

JAVA后端开发——“全量同步”和“增量同步”

“全量同步”和“增量同步”是数据处理、系统集成和数据库领域中两个基本概念。描述了两种截然不同的数据同步策略&#xff0c;理解它们的区别对于设计任何数据系统都至关重要。全量同步 核心思想&#xff1a;全部替换&#xff0c;一步到位。在技术上&#xff0c;全量同步通常意…

修改CentOS的SSH登录端口(22端口)

要修改CentOS系统的SSH服务默认端口(22端口)&#xff0c;请按照以下步骤操作&#xff1a; 备份SSH配置文件 sudo cp /etc/ssh/sshd_config /etc/ssh/sshd_config.bak编辑SSH配置文件 sudo vi /etc/ssh/sshd_config查找并修改端口设置 找到以下行(大约在第13行左右)&#xff1a;…

python导包机制-更优方式

在学习某个大模型应用的后端时&#xff0c;发现&#xff1a; xxx |-----src |------\---modules |------\------\------b.py |-----app.py在app.py中可以使用src.modules.b来进行导入。之前我导入时是形如.modules.b这种形式&#xff08;前面有.&#xff09;&#xff0c;但是当…

检索召回率优化探究一:基于 LangChain 0.3集成 Milvus 2.5向量数据库构建的智能问答系统

背景 基于 LangChain 0.3集成 Milvus 2.5向量数据库构建的 NFRA&#xff08;National Financial Regulatory Administration&#xff0c;国家金融监督管理总局&#xff09;政策法规智能问答系统&#xff0c;第一个版本的检索召回率是 79.52%&#xff0c;尚未达到良好、甚至是优…

《整合Spring Cache:本地缓存、Redis与Caffeine对比实践》

&#x1f680; 整合Spring Cache&#xff1a;本地缓存、Redis与Caffeine对比实践 &#x1f4cc; 前言 在高并发、高性能的系统设计中&#xff0c;缓存始终扮演着不可替代的角色。Spring Cache 作为 Spring 框架原生提供的缓存抽象层&#xff0c;极大简化了缓存接入的复杂度。…

easyexcel填充方式导出-合并单元格并设置边框

填充的模板最后导出效果实体 /*** 账户实体类* author test* date 2025-07-28*/ Getter Setter class Test {/*** 账户类型*/private String accType;/*** 账户余额*/private String money; }导出逻辑 /*** 导出文件逻辑*/ public void exportReport(List<Test> data) { …

Jenkins + SonarQube 从原理到实战一:基于 K8s 部署与使用(含中文插件与 Python 扫描)

前言 公司开发部门希望在 Jenkins 构建过程中自动集成 C/C 的代码扫描&#xff0c;正好我也没接触过 SonarQube&#xff0c;于是记录下从零开始部署 SonarQube 服务并集成到 CI/CD 的过程&#xff0c;供后来者参考。 一、SonarQube 原理与工作机制详解 1.1 什么是 SonarQube&…

Linux(Centos 7.6)命令详解:sz

1.命令作用使用ZMODEM/YMODEM/XMODEM协议发送文件(Send file(s) with ZMODEM/YMODEM/XMODEM protocol)注意: 需要yum install lrzsz (yum provides sz可以查看rz命令是什么rpm包提供的)2.命令语法Usage: sz [options] file ...or: sz [options] -{c|i} COMMAND3.参数详解OPTION…

智能运维中的数据转换

《智能运维实践 苏娜 孙琳 王鸽著 人工智能技术丛书 自然语言处理的常用算法 日志异常检测 根因定位 网络流量异常检测 清华大学出版社》【摘要 书评 试读】- 京东图书 数据转换是数据预处理中的关键步骤&#xff0c;用于将数据从原始格式转换为适合分析和建模的形式。这一过程…

IAR编辑器如何让左侧的工具栏显示出来?

在IAR编辑器中恢复左侧工具栏显示&#xff0c;可通过以下方法操作&#xff1a; 一、通过菜单栏启用工具栏 ‌进入视图菜单‌ 点击顶部菜单栏的 ‌"View"‌ → 在弹出列表中勾选 ‌"Workspace"‌ 若工具栏仍不显示&#xff0c;查看菜单栏右侧是否有 ‌"…

ADB+Python控制(有线/无线) Scrcpy+按键映射(推荐)

要实现电脑通过键盘控制安卓平板屏幕点击的功能&#xff0c;可以采用以下方案&#xff1a; 方案一&#xff1a;ADBPython控制&#xff08;有线/无线&#xff09; 准备工具&#xff1a; 安卓平板开启开发者模式&#xff08;设置→关于平板→连续点击版本号&#xff09;启用USB调…

同态滤波算法详解:基于频域变换的光照不均匀校正

&#x1f3ad; 同态滤波&#xff1a;图像频域的调音师技术“如同调音师在音频处理中分离并调节不同频率成分&#xff0c;同态滤波能够在图像频域中精确分离光照与细节信息。”&#x1f3af; 图像频域调音的技术挑战 在数字图像处理中&#xff0c;光照不均匀问题如同音频中的混响…

Ubuntu简述及部署系统

1.什么是Ubuntu1.1概述Ubuntu属于Debian系列&#xff0c;Debian是社区类Linux的典范&#xff0c;是迄今为止最遵循GNU规范的Linux系统。Debain最早由lan Murdock于1993年创建&#xff0c;分为三个版本分支&#xff08;branch&#xff09;&#xff1a;stable&#xff0c;testing…

Claude Code安装部署

1️⃣安装 Node.js&#xff08;已安装可跳过&#xff09; 确保 Node.js 版本 ≥ 18.0 # Ubuntu / Debian 用户 curl -fsSL https://deb.nodesource.com/setup_lts.x | sudo bash - sudo apt-get install -y nodejs node --version# macOS 用户 sudo xcode-select --install /b…