代码随想录算法训练营 Day32 | 动态规划 part05

news2026/4/16 10:12:19
52. 携带研究材料第七期模拟笔试题目描述小明是一位科学家他需要参加一场重要的国际科学大会以展示自己的最新研究成果。他需要带一些研究材料但是他的行李箱空间有限。这些研究材料包括实验设备、文献资料和实验样本等等它们各自占据不同的重量并且具有不同的价值。小明的行李箱所能承担的总重量是有限的问小明应该如何抉择才能携带最大价值的研究材料每种研究材料可以选择无数次并且可以重复选择。输入描述第一行包含两个整数nv分别表示研究材料的种类和行李所能承担的总重量接下来包含 n 行每行两个整数 wi 和 vi代表第 i 种研究材料的重量和价值输出描述输出一个整数表示最大价值。输入示例4 51 22 43 44 5输出示例10#include iostream #include vector using namespace std; int main(){ int n, bagSize; cin n bagSize; vectorint w(n 1); vectorint v(n 1); // 读入物品的重量和价值您的代码从下标 0 开始读入完全没问题 for(int i 0; i n; i) cin w[i] v[i]; // 1. 定义 DP 数组 // dp[j] 表示容量为 j 的背包能装下的最大价值 vectorint dp(bagSize 1, 0); // 2. 遍历物品 for(int i 0; i n; i){ // 3. 遍历背包容量 —— 核心变化正序遍历 for(int j 0; j bagSize; j){ if(j w[i]){ // 递推公式与 0-1 背包一模一样 // 区别全在遍历顺序上 dp[j] max(dp[j], dp[j - w[i]] v[i]); } } } cout dp[bagSize]; return 0; }总结1. 为什么正序遍历就能实现“无限次使用”以计算dp[5]且当前物品重量w[i] 2为例0-1 背包逆序计算dp[5]时需要用dp[3]。因为逆序dp[3]还没被更新它代表没有放当前物品时的状态。所以当前物品只能放 1 次。完全背包正序计算dp[5]时需要用dp[3]。因为正序dp[3]已经被更新过了。此时的dp[3]已经包含了当前物品所以dp[5] dp[3] v[i]相当于在dp[3]的基础上又放了一次当前物品。这样就实现了物品的重复选取。2. 优化在完全背包中我们可以直接把j的起始点设为w[i]省略掉if判断代码会更简洁for(int i 0; i n; i){ // 直接从 w[i] 开始正序遍历 for(int j w[i]; j bagSize; j){ dp[j] max(dp[j], dp[j - w[i]] v[i]); } }3. 纯背包问题的总结对照表特性0-1 背包完全背包物品数量只能选 1 次可以选无数次一维数组遍历顺序逆序 (j bagSize - w[i])正序 (j w[i] - bagSize)递推公式dp[j] max(dp[j], dp[j-w] v)dp[j] max(dp[j], dp[j-w] v)518. 零钱兑换 II给你一个整数数组coins表示不同面额的硬币另给一个整数amount表示总金额。请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额返回0。假设每一种面额的硬币有无限个。题目数据保证结果符合 32 位带符号整数。class Solution { public: int change(int amount, vectorint coins) { int n coins.size(); // 1. 定义 DP 数组 // dp[j] 表示凑成总金额 j 的硬币组合数 // 【高光细节】这里必须用 uint64_t 或者 long long // 因为组合数会非常大普通的 int 会在 LeetCode 的测试用例中溢出导致错误 vectoruint64_t dp(amount 1, 0); // 2. 初始化 // 凑成金额 0 的组合数是 1即“什么都不选”这一种方法 dp[0] 1; // 3. 状态转移 // 外层遍历物品硬币面额 for(int i 0; i n; i){ // 内层遍历背包容量目标金额 // 完全背包正序遍历从当前硬币面额开始 for(int j coins[i]; j amount; j){ // 递推公式求组合数累加方案数 // dp[j] 不用当前硬币的组合数 使用当前硬币的组合数 dp[j] dp[j - coins[i]]; } } return (int)dp[amount]; } };总结1. 为什么外层必须是物品内层是背包这是求 组合数 和求 排列数 的核心区别。外层物品内层背包 - 求组合数假设coins [1, 5]。当外层循环固定是1时内层循环会把所有包含1的组合算出来如[1,1,1]。当外层循环走到5时它只能加在之前算出的结果后面如[1,1,1,5]。结果5永远在1的后面。算出来的是 组合[1,5] 和 [5,1] 算同一种。外层背包内层物品 - 求排列数假设coins [1, 5]目标金额是 6。当外层循环走到金额6时内层循环先遇到1算出[...1]接着遇到5算出[...5]。如果在某次循环中先凑出了 5下一次金额 6 循环时先遇到 1就会变成[5, 1]。结果1和5的顺序可以颠倒。算出来的是 排列。2.uint64_t的妙用在 C 中int最大只能表示约 21 亿。对于稍微大一点的amount组合数会呈指数级增长。使用uint64_t(无符号 64 位整数) 完美避开了溢出报错最后强转回int返回。3. 复杂度分析时间复杂度O(N×M)N 是硬币种类数M 是目标金额。空间复杂度O(M)。377. 组合总和 Ⅳ给你一个由 不同 整数组成的数组nums和一个目标整数target。请你从nums中找出并返回总和为target的元素排列的个数。题目数据保证答案符合 32 位整数范围。class Solution { public: int combinationSum4(vectorint nums, int target) { int n nums.size(); // 1. 定义 DP 数组 // dp[j] 表示凑成目标和为 j 的排列总数 // 依然使用 uint64_t 防止极端用例下的整型溢出 vectoruint64_t dp(target 1, 0); // 2. 初始化 // 凑成目标和为 0 的排列数是 1什么都不选 dp[0] 1; // 3. 状态转移 // 【核心考点】外层遍历背包目标内层遍历物品数字 for(int j 0; j target; j){ for(int i 0; i n; i){ // 防止数组越界 if(j nums[i]){ // 递推公式与求组合数完全一样累加方案数 dp[j] dp[j - nums[i]]; } } } return dp[target]; } };总结1. 题目解析为什么叫“组合”却求“排列”LeetCode 这道题的名字极具误导性。题目描述中说“不同的序列被视为不同的组合”。在正常的数学定义中[1, 3]和[3, 1]是同一种组合但属于两种不同的排列。题目既然把不同顺序算作不同结果那么它本质上求的就是 排列数。2. 遍历顺序外层j内层i代码求排列外层是target。当我们要计算dp[6]时我们会依次把nums里的数放进去试。如果nums [1, 2, 3]计算dp[6]时先加dp[5]意味着排列的最后一步是 1再加dp[4]意味着排列的最后一步是 2再加dp[3]意味着排列的最后一步是 3效果1可以在前面也可以在后面元素的先后顺序被完全保留了算出来的是 排列数。上一题代码求组合外层是nums。当外层固定是1时所有的1都只能被最先放入背包。效果1永远排在2和3的前面打乱了原始顺序算出来的是 组合数。公式都是dp[j] dp[j - nums[i]]仅仅是for循环换了个位置结果就截然不同这就是动态规划的美丽与恐怖之处。3.if(j nums[i])的必要性在求组合数时我们可以直接把内层循环写成for(int j coins[i]; j amount; j)省去if判断。但在求排列数时不能这么写。因为外层循环是在遍历j而nums[i]在内层每次i变化时nums[i]都不同所以必须在内部用if来做越界保护。您的处理非常严谨。57. 爬楼梯第八期模拟笔试题目描述假设你正在爬楼梯。需要 n 阶你才能到达楼顶。每次你可以爬至多m (1 m n)个台阶。你有多少种不同的方法可以爬到楼顶呢注意给定 n 是一个正整数。输入描述输入共一行包含两个正整数分别表示n, m输出描述输出一个整数表示爬到楼顶的方法数。输入示例3 2输出示例3#include iostream #include vector using namespace std; int main(){ int n, m; cin n m; // n: 目标台阶数(背包容量), m: 一次最多爬的步数(物品范围) // 1. 定义 DP 数组 // dp[j] 表示爬到第 j 阶有多少种不同的排列方法 vectorint dp(n 1, 0); // 2. 初始化 // 爬到第 0 阶起点有一种方法就是原地不动 dp[0] 1; // 3. 状态转移 // 【核心】求排列数外层必须是背包容量台阶 j内层是物品步数 i for(int j 1; j n; j){ for(int i 1; i m; i){ if(j i){ // 递推公式累加方案数 // 爬到 j 阶的方法数 爬到 j-i 阶的方法数 dp[j] dp[j - i]; } } } cout dp[n]; return 0; }总结1. 为什么必须是“外层 j内层 i”如果反过来写外层i内层j比如m2外层i1时算出的全是以1开头的序列如1,1,1...或1,2,1...。外层i2时算出的全是以2开头的序列如2,1,1...。这就变成了 组合数先走 1 后走 2和先走 2 后走 1 被当成同一种。但爬楼梯显然讲究顺序先跨 1 步再跨 2 步和先跨 2 步再跨 1 步是两种不同的爬法所以必须用 外层 j 内层 i 来求排列数。2. 与纯背包题的细微差别纯背包题给你一个数组如[1, 5, 2]数组里的元素是固定的可能无序。本题隐含的数组是[1, 2, 3, ..., m]这是一个连续递增的序列。但不管物品是乱序还是连续递增只要是求排列数模板就绝对不能变。

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

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

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…