代码随想录算法训练营第三十天 | 卡码网46.携带研究材料(二维解法)、卡码网46.携带研究材料(滚动数组)、LeetCode416.分割等和子集

代码随想录算法训练营第三十天 | 卡码网46.携带研究材料(二维解法)、卡码网46.携带研究材料(滚动数组)、LeetCode416.分割等和子集

01-1 卡码网46.携带研究材料(二维)

相关资源

  • 题目链接:46. 携带研究材料

  • 文章讲解:01背包二维问题

  • 视频讲解:0-1背包问题

题目:

小明是一位科学家,他需要参加一场重要的国际科学大会,以展示自己的最新研究成果。他需要带一些研究材料,但是他的行李箱空间有限。这些研究材料包括实验设备、文献资料和实验样本等等,它们各自占据不同的空间,并且具有不同的价值。

小明的行李空间为 N,问小明应该如何抉择,才能携带最大价值的研究材料,每种研究材料只能选择一次,并且只有选与不选两种选择,不能进行切割。

第一想法:我之前是接触过01背包问题的,但现在一点思路也没有

看完代码随想录之后的想法

五部曲

  1. dp数组的含义:dp[i][j]表示从下标为[0-i]的物品里任意取,放进容量为j的背包,价值总和最大是多少
  2. 确定递推公式:dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i])
  3. 初始化dp数组,dp[i][0]=0,当 j < weight[0]的时候,dp[0][j] 是 0,因为背包容量比编号0的物品重量还小。当j >= weight[0]时,dp[0][j]应该是value[0],因为背包容量放足够放编号0物品。
  4. 确定遍历顺序,先遍历物品和先遍历背包重量都可以,因为均由左上角推导得到
  5. 打印dp数组

实现

#include <bits/stdc++.h>
using namespace std;int main() {int n, bagweight;// bagweight代表行李箱空间cin >> n >> bagweight;vector<int> weight(n, 0); // 存储每件物品所占空间vector<int> value(n, 0);  // 存储每件物品价值for(int i = 0; i < n; ++i) {cin >> weight[i];}for(int j = 0; j < n; ++j) {cin >> value[j];}// dp数组, dp[i][j]代表行李箱空间为j的情况下,从下标为[0, i]的物品里面任意取,能达到的最大价值vector<vector<int>> dp(weight.size(), vector<int>(bagweight + 1, 0));// 初始化, 因为需要用到dp[i - 1]的值// j < weight[0]已在上方被初始化为0// j >= weight[0]的值就初始化为value[0]for (int j = weight[0]; j <= bagweight; j++) {dp[0][j] = value[0];}for(int i = 1; i < weight.size(); i++) { // 遍历科研物品for(int j = 0; j <= bagweight; j++) { // 遍历行李箱容量if (j < weight[i]) dp[i][j] = dp[i - 1][j]; // 如果装不下这个物品,那么就继承dp[i - 1][j]的值else {dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);}}}cout << dp[n - 1][bagweight] << endl;return 0;
}

ToDo:自己写出来

2025-3-3:

#include <bits/stdc++.h>
using namespace std;
int main(){int M, N;cin >> M >> N;vector<int> weight(M,0);vector<int> value(M,0);for(int i = 0; i < M; i++){cin >> weight[i];}for(int j = 0; j < M; j++){cin >> value[j];}// 初始化vector<vector<int>> dp(weight.size(),vector<int>(N+1,0));for(int j = weight[0]; j<= N; j++){dp[0][j] = value[0];} for(int i = 1; i < M; i++){for(int j = 0; j <= N; j++){if(j<weight[i]) dp[i][j]=dp[i-1][j];else{dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);}}}cout << dp[M - 1][N] << endl;
}
01-2 卡码网46.携带研究材料(一维)

相关资源

  • 题目链接:46. 携带研究材料

  • 文章讲解:01背包问题一维

  • 视频讲解:01背包问题(滚动数组篇)

题目:

小明是一位科学家,他需要参加一场重要的国际科学大会,以展示自己的最新研究成果。他需要带一些研究材料,但是他的行李箱空间有限。这些研究材料包括实验设备、文献资料和实验样本等等,它们各自占据不同的空间,并且具有不同的价值。

小明的行李空间为 N,问小明应该如何抉择,才能携带最大价值的研究材料,每种研究材料只能选择一次,并且只有选与不选两种选择,不能进行切割。

看完代码随想录之后的想法

五部曲

  1. dp数组的含义:dp[j]表示:容量为j的背包,所背的物品价值可以最大为dp[j]
  2. 确定递推公式:dp[j] = max(dp[j], dp[j - weight[i]] + value[i])
  3. 初始化dp数组,dp数组初始化的时候,都初始为0就可以了
  4. 确定遍历顺序,先遍历物品再遍历背包重量,并且要求背包容量由大到小遍历,从而保证物品i只被放入一次
  5. 打印数组

实现

// 一维dp数组实现
#include <iostream>
#include <vector>
using namespace std;int main() {// 读取 M 和 Nint M, N;cin >> M >> N;vector<int> costs(M);vector<int> values(M);for (int i = 0; i < M; i++) {cin >> costs[i];}for (int j = 0; j < M; j++) {cin >> values[j];}// 创建一个动态规划数组dp,初始值为0vector<int> dp(N + 1, 0);// 外层循环遍历每个类型的研究材料for (int i = 0; i < M; ++i) {// 内层循环从 N 空间逐渐减少到当前研究材料所占空间for (int j = N; j >= costs[i]; --j) {// 考虑当前研究材料选择和不选择的情况,选择最大值dp[j] = max(dp[j], dp[j - costs[i]] + values[i]);}}// 输出dp[N],即在给定 N 行李空间可以携带的研究材料最大价值cout << dp[N] << endl;return 0;
}

ToDo:复刻

2025-3-3:

#include <bits/stdc++.h>
using namespace std;
int main(){int M, N;cin >> M >> N;vector<int> weight(M,0);vector<int> value(M,0);for(int i = 0; i < M; i++){cin >> weight[i];}for(int j = 0; j < M; j++){cin >> value[j];}// 初始化vector<int> dp(N+1,0);for(int i = 0; i < M; i++){for(int j = N; j >= weight[i]; j--){dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);}}cout << dp[N] << endl;
}

感悟:其实从前往后还是从后往前遍历很好想,二维遍历过程可以发现,当前的dp[i][j] 来自于上方正对和上方偏左,如果一维滚动数组从前往后遍历,前面的发生改变,后面的就会被影响;至于为什么先遍历物品,再遍历背包容量,假如顺序颠倒过来,其实可以还原成二维遍历过程,因为二维遍历的从上往下再从左往右,当前单元格其实是用到了当前单元格的上一个单元格,并不能竖着拷贝(要想这样,除非dp[i][j] = max(dp[i][j-1], dp[i - weight[j]+value[j]][j - 1),换句话说i和j是不可对调等价的)

01-3 LeetCode416.分割等和子集

相关资源

  • 题目链接:416. 分割等和子集

  • 文章讲解:分割等和子集

  • 视频讲解:LeetCode:416.分割等和子集

题目:

给你一个 只包含正整数非空 数组 nums 。请你判断是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。

第一想法:这个想不出来和01背包有关系

看完代码随想录之后的想法

问题转化:

首先,本题要求集合里能否出现总和为 sum / 2 的子集。既有一个 只能装重量为 sum / 2 的背包,商品为数字,这些数字能不能把 这个背包装满。那每一件商品是数字的话,对应的重量 和 价值是多少呢?一个数字只有一个维度,即 重量等于价值。当数字 可以装满 承载重量为 sum / 2 的背包的背包时,这个背包的价值也是 sum / 2。那么这道题就是 装满 承载重量为 sum / 2 的背包,价值最大是多少?如果最大价值是 sum / 2,说明正好被商品装满了

动规五部曲分析如下:

  1. 确定dp数组以及下标的含义:01背包中,dp[j] 表示: 容量(所能装的重量)为j的背包,所背的物品价值最大可以为dp[j]

  2. 确定递推公式:dp[j] = max(dp[j], dp[j - nums[i]] + nums[i])

  3. 初始化:为0

  4. 遍历顺序:一维dp数组,物品遍历的for循环放在外层,遍历背包的for循环放在内层,且内层for循环倒序遍历

  5. 打印数组

实现:

class Solution {
public:bool canPartition(vector<int>& nums) {int sum = 0;// dp[i]中的i表示背包内总和// 题目中说:每个数组中的元素不会超过 100,数组的大小不会超过 200// 总和不会大于20000,背包最大只需要其中一半,所以10001大小就可以了vector<int> dp(10001, 0);for (int i = 0; i < nums.size(); i++) {sum += nums[i];}// 也可以使用库函数一步求和// int sum = accumulate(nums.begin(), nums.end(), 0);if (sum % 2 == 1) return false;int target = sum / 2;// 开始 01背包for(int i = 0; i < nums.size(); i++) {for(int j = target; j >= nums[i]; j--) { // 每一个元素一定是不可重复放入,所以从大到小遍历dp[j] = max(dp[j], dp[j - nums[i]] + nums[i]);}}// 集合中的元素正好可以凑成总和targetif (dp[target] == target) return true;return false;}
};

收获:题目中物品是nums[i],重量是nums[i],价值也是nums[i],背包体积是sum/2。

ToDo:复现

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

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

相关文章

nvidia驱动更新,centos下安装openwebui+ollama(非docker)

查看centos内核版本 uname -a cat /etc/redhat-release下载对应的程序&#xff08;这个是linux64位版本通用的&#xff09; https://cn.download.nvidia.cn/tesla/550.144.03/NVIDIA-Linux-x86_64-550.144.03.run cudnn想办法自己下一下&#xff0c;我这里是12.x和11.x通用的…

【AIGC系列】4:Stable Diffusion应用实践和代码分析

AIGC系列博文&#xff1a; 【AIGC系列】1&#xff1a;自编码器&#xff08;AutoEncoder, AE&#xff09; 【AIGC系列】2&#xff1a;DALLE 2模型介绍&#xff08;内含扩散模型介绍&#xff09; 【AIGC系列】3&#xff1a;Stable Diffusion模型原理介绍 【AIGC系列】4&#xff1…

51单片机-串口通信编程

串行口工作之前&#xff0c;应对其进行初始化&#xff0c;主要是设置产生波特率的定时器1、串行口控制盒中断控制。具体步骤如下&#xff1a; 确定T1的工作方式&#xff08;编程TMOD寄存器&#xff09;计算T1的初值&#xff0c;装载TH1\TL1启动T1&#xff08;编程TCON中的TR1位…

Windows 10 远程桌面连接使用指南

目录 一、引言 二、准备工作 1、确认系统版本 2、服务器端设置 三、客户端连接 1、打开远程桌面连接程序 2、输入连接信息 3、输入登录凭证 4、开始使用远程桌面 四、移动端连接&#xff08;以 iOS 为例&#xff09; 1、下载安装应用 2、添加远程计算机 3、进行连接…

spring boot打包插件的问题

在spring boot项目中声明了 <build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin></plugins></build> 执行mvn clean package&…

R语言+AI提示词:贝叶斯广义线性混合效应模型GLMM生物学Meta分析

全文链接&#xff1a;https://tecdat.cn/?p40797 本文旨在帮助0基础或只有简单编程基础的研究学者&#xff0c;通过 AI 的提示词工程&#xff0c;使用 R 语言完成元分析&#xff0c;包括数据处理、模型构建、评估以及结果解读等步骤&#xff08;点击文末“阅读原文”获取完整代…

iOS UICollectionViewCell 点击事件自动化埋点

iOS 中经常要进行埋点&#xff0c;我们这里支持 UICollectionViewCell. 进行自动化埋点&#xff0c;思路&#xff1a; 通过hook UICollectionViewCell 的setSelected:方法&#xff0c; 则新的方法中执行埋点逻辑&#xff0c;并调用原来的方法 直接上代码 implementation UICol…

课程《MIT Introduction to Deep Learning》

在Youtubu上&#xff0c;MIT Introduction to Deep Learning (2024) | 6.S191 共8节课&#xff1a; (1) MIT Introduction to Deep Learning (2024) | 6.S191 (2) MIT 6.S191: Recurrent Neural Networks, Transformers, and Attention (3) MIT 6.S191: Convolutional Neural N…

Docker 学习(一)

一、Docker 核心概念 Docker 是一个开源的容器化平台&#xff0c;允许开发者将应用及其所有依赖&#xff08;代码、运行时、系统工具、库等&#xff09;打包成一个轻量级、可移植的“容器”&#xff0c;实现 “一次构建&#xff0c;随处运行”。 1、容器&#xff08;Container…

007 订单支付超时自动取消订单(rabbitmq死信队列 mybatis)

文章目录 死信队列RabbitMQ 配置类 RabbitMQConfig.java生产者 OrderTimeoutProducer.java消费者 OrderTimeoutConsumer.java应用配置 application.ymlpom.xml 依赖实体类 Order.java&#xff08;不变&#xff09;Mapper 接口 OrderMapper.java&#xff08;不变&#xff09;服务…

计算机毕业设计SpringBoot+Vue.js智慧图书管理系统(源码+文档+PPT+讲解)

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…

《论数据分片技术及其应用》审题技巧 - 系统架构设计师

论数据分片技术及其应用写作框架 一、考点概述 本论题“论数据分片技术及其应用”主要考察的是软件工程中数据分片技术的理解、应用及其实际效果分析。考点涵盖以下几个方面&#xff1a; 首先&#xff0c;考生需对数据分片的基本概念有清晰的认识&#xff0c;理解数据分片是…

【每日学点HarmnoyOS Next知识】web加载pdf、Toggle禁用、Grid多次渲染问题、Web判断是否存在title、 List侧滑栏关闭

【每日学点HarmnoyOS Next知识】web加载pdf、Toggle禁用、Grid多次渲染问题、Web判断是否存在title、 List侧滑栏关闭 1、HarmonyOS Web组件加载本地pdf文件后&#xff0c;默认显示标题和下载按钮&#xff0c;可以隐藏或者有对应的操作这个title的API吗&#xff1f; 隐藏PDF操…

下载 MindSpore 配置 PyTorch环境

以下是下载 MindSpore 并配置 PyTorch 环境的详细步骤&#xff0c;适用于常见的 Linux/Windows 系统&#xff08;以 NVIDIA GPU 为例&#xff09;&#xff1a; 一、环境准备 1. 硬件与软件检查 GPU 支持&#xff1a;确保使用 NVIDIA 显卡&#xff0c;通过 nvidia-smi 查看驱动…

三、数据提取

利用 requests 可以获取网站页面数据&#xff0c;但是 requests 返回的数据中包含了一些冗余数据&#xff0c;我们需要在这些数据集中提取自己需要的信息。所以我们要学会在数据集中提取自己需要的数据。 需要掌握的知识点如下&#xff1a; json 数据提取 jsonpath 语法 静态…

Qt | 实战继承自QObject的IOThread子类实现TCP客户端(安全销毁)

点击上方"蓝字"关注我们 01、QThread >>> start() 启动线程,调用后会执行 run() 方法。 run() 线程的入口点,子类化 QThread 时需要重写此方法以定义线程的执行逻辑。 quit() 请求线程退出,线程会在事件循环结束后终止。 exit(int returnCode = 0) 退出…

int new_pos = (pos + delta + 9) % 9 化曲为直算法

公式 int new_pos (pos delta 9) % 9; 是一个常见的 循环数组索引计算 方法&#xff0c;用于处理圆圈排列中的位置计算。这个公式可以总结出一个普遍的规律&#xff0c;适用于任何循环数组或圆圈排列的场景。 普遍规律 假设有一个长度为 ( n ) 的循环数组&#xff08;或圆圈…

生成一个日期时间序列,从‘2024-12-03‘开始,每小时递增 oracle 转为达梦

-------------------------------生成一个日期时间序列&#xff0c;从2024-12-03开始&#xff0c;每小时递增---------------------------- ---原oracle : SELECT to_date(2024-12-03, yyyy-mm-dd) (ROWNUM - 1) / 24 data_time FROM dual CO…

前端学习——HTML

VSCode常用快捷键 代码格式化&#xff1a;ShiftAltF 向上或向下移动一行&#xff1a;AltUp或AltDown 快速复制一行代码&#xff1a;ShiftAltUp或者ShiftAltDown 快速替换&#xff1a;CtrlH HTML标签 文本标签 定义着重文字 定义粗体文字 定义斜体文字 加重语气 删除字 无特…

Hadoop之02:MR-图解

1、不是所有的MR都适合combine 1.1、map端统计出了不同班级的每个学生的年龄 如&#xff1a;(class1, 14)表示class1班的一个学生的年龄是14岁。 第一个map任务&#xff1a; class1 14 class1 15 class1 16 class2 10第二个map任务&#xff1a; class1 16 class2 10 class…