【卡码网】完全背包问题 52. 携带研究材料——代码随想录算法训练营Day44

题目链接:题目页面

题目描述

题目描述

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

小明的行李箱所能承担的总重量为 N,问小明应该如何抉择,才能携带最大价值的研究材料,每种研究材料可以选择无数次,并且可以重复选择。

输入描述

第一行包含两个整数,N,V,分别表示研究材料的种类和行李空间 

接下来包含 N 行,每行两个整数 wi 和 vi,代表第 i 种研究材料的重量和价值

输出描述

输出一个整数,表示最大价值。

输入示例
4 5
1 2
2 4
3 4
4 5
输出示例
10
提示信息

第一种材料选择五次,可以达到最大值。

数据范围:

1 <= N <= 10000;
1 <= V <= 10000;
1 <= wi, vi <= 10^9.

文章讲解:代码随想录

视频讲解:带你学透完全背包问题! 和 01背包有什么差别?遍历顺序上有什么讲究?_哔哩哔哩_bilibili

题解1:动态规划

思路:类似01背包问题,完全背包问题也可以用动态规划法求解。完全背包和01背包的区别在于每个物品可以取任意次。

动态规划分析:

  • dp 数组以及下标的含义:dp[i][j] 表示从下标为0到 i 的物品里任意取,放进容量为 j 的背包,价值总和最大是多少。
  • 递推公式:由于每个元素能取多次,而 dp[i][j] 的状态依赖于当前元素一个也不取和再取1个当前元素。当前元素一个也不取即 dp[i - 1][j],再取一个当前元素即 dp[i][j - weight[i]] + value[i]。即 dp[i][j] = Math.max(dp[i - 1][j], dp[i][j - weight[i]] + value[i])。
  • dp 数组初始化:dp[i][j] 的状态依赖于正上方和正左方的状态,因此需要初始化第0行和第0列,即 dp[0, j] 和 dp[i, 0]。dp[i, 0] 表示从下标为0到 i 的物品里任意取,放进容量为 j 的背包,价值总和最大自然是0。dp[0, j] 表示取或者不取下标为0的物品,放进容量为 j 的背包,j 小于 weight[0] 时,价值总和最大为0,大于等于 weight[0] 时为 k 倍value[0],k 取整数且最大满足 k * weight[0] < j。
  • 遍历顺序:dp[i][j] 的状态依赖于正上方和正左方的状态,因此在填充 dp[i][j] 时,它的正上方和同行的正左方需要填充。先遍历物品再遍历背包和先遍历背包再遍历物品这两种方式都可以。
  • 打印 dp 数组:以如下输入为例

4 5
1 2
2 4
3 4
4 5

dp 数组为 [ [ 0, 2, 4, 6, 8, 10 ],  [ 0, 2, 4, 6, 8, 10 ],  [ 0, 2, 4, 6, 8, 10 ],  [ 0, 2, 4, 6, 8, 10 ] ]。

const readline = require('readline');
const rl = readline.createInterface({input: process.stdin,output: process.stdout
});let num = 1;
const inputs = [];rl.on("line", (row) => {inputs.push(row);if (inputs.length === 1) {num += parseInt(inputs[0]);}if (inputs.length < num) {return;}const firstRow = inputs[0].split(' ');const n = parseInt(firstRow[0]);const w = parseInt(firstRow[1]);const space = [];const value = [];for (let i = 0; i < n; i++) {[space[i], value[i]] = inputs[i + 1].split(" ").map(s => parseInt(s));}// 定义 dp 数组const dp = new Array(n).fill().map(() => new Array(w + 1).fill(0));// 初始化 dp 数组for (let j = 0; j < w + 1; j++) {let totalSpace = 0;while (j - totalSpace >= space[0]) {totalSpace += space[0];dp[0][j] += value[0];}}// 先遍历物品,再遍历背包for (let i = 1; i < n; i++) {for (let j = 1; j < w + 1; j++) {if (j < space[i]) {dp[i][j] = dp[i - 1][j];} else {dp[i][j] = Math.max(dp[i - 1][j], dp[i][j - space[i]] + value[i]);}}}console.log(dp[n - 1][w]);
});

分析:令物品数量为 n,背包最大容量为 w,则时间复杂度为 O(n * w),空间复杂度为 O(n * w)。

题解2:动态规划优化

思路:dp[i][j] 依赖于上一行正上方及本行正左方的状态,与同一行后面的状态无关。可以想到填充某一行时,将上一行内容覆盖到这一行,上一行正上方即本身,这样 dp[j] 的状态只依赖于前面的状态。

动态规划分析:

  • dp 数组以及下标的含义:dp[j]表示容量为j的背包,所背的物品价值可以最大为dp[j]。
  • 递推公式:dp[j] = Math.max(dp[j], dp[j - weight[i]] + value[i])。
  • dp 数组初始化:dp[0] 为0,j 大于0时 dp[j] 依赖于上一轮的 dp[j] 和前面的状态,为了取最大值后结果正确,应该初始化为0。即全部初始化为0。
  • 遍历顺序:dp[j] 的状态依赖于正左方的状态,因此在填充 dp[j] 时,它的正左方需要填充。先遍历物品再遍历背包和先遍历背包再遍历物品这两种方式都可以。
  • 打印 dp 数组:以如下输入为例

4 5
1 2
2 4
3 4
4 5

每一层的 dp 数组为

[ 0, 2, 4, 6, 8, 10 ]
[ 0, 2, 4, 6, 8, 10 ]
[ 0, 2, 4, 6, 8, 10 ]
[ 0, 2, 4, 6, 8, 10 ]

可以看到,将每一层的 dp 数组结合起来,和二维 dp 数组相同。

const readline = require('readline');
const rl = readline.createInterface({input: process.stdin,output: process.stdout
});let num = 1;
const inputs = [];rl.on("line", (row) => {inputs.push(row);if (inputs.length === 1) {num += parseInt(inputs[0]);}if (inputs.length < num) {return;}const firstRow = inputs[0].split(' ');const n = parseInt(firstRow[0]);const w = parseInt(firstRow[1]);const space = [];const value = [];for (let i = 0; i < n; i++) {[space[i], value[i]] = inputs[i + 1].split(" ").map(s => parseInt(s));}// 定义 dp 数组const dp = new Array(w + 1).fill(0);// 先遍历物品,再遍历背包for (let i = 0; i < n; i++) {// 正序遍历背包for (let j = space[i]; j <= w; j++) {dp[j] = Math.max(dp[j], dp[j - space[i]] + value[i]);}}console.log(dp[w]);
});

分析:令物品数量为 n,背包最大容量为 w,则时间复杂度为 O(n * w),空间复杂度为 O(w)。

收获

学习使用动态规划法求解完全背包的理论知识。

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

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

相关文章

终于明白kmp算法

在刷代码随想录的时候&#xff0c;遇到了leetcode这道经典题目 28. 实现 strStr() 力扣题目链接(opens new window) 实现 strStr() 函数。 给定一个 haystack 字符串和一个 needle 字符串&#xff0c;在 haystack 字符串中找出 needle 字符串出现的第一个位置 (从0开始)。如…

Selenium IDE插件录制网页,解放双手

1、 国内下载地址 https://www.crx4chrome.com/crx/77585/ &#xff0c;这个网络正常基本可以下载&#xff0c;目前最新版本是3.17.2。 点击Crx4Chrome下载。下载后的文件名称是&#xff1a;mooikfkahbdckldjjndioackbalphokd-3.17.2-Crx4Chrome.com.crx。 2、 安装 直接打开…

webpack 如何实现模块懒加载

首先在打包的时候使用 splitwebpackplugin 进行分割 在实际引用中&#xff0c;webpack 实现模块的懒加载是通过动态导入&#xff08;dynamic import &#xff09;来实现的。动态导入是 es6 的一项功能&#xff0c;允许在运行时异步加载模块&#xff0c;从而实现按需加载。 使…

CMS垃圾回收器

CMS垃圾回收 CMS GC的官方名称为“Mostly Concurrenct Mark and Sweep Garbage Collector”&#xff08;最大-并发-标记-清除-垃圾收集器&#xff09;。 作用范围&#xff1a; 老年代 算法&#xff1a; 并发标记清除算法。 启用参数&#xff1a;-XX:UseConMarkSweepGC 默认回收…

探索创造无限可能——Autodesk AutoCAD 2022(CAD 2022)系统要求

随着科技的不断进步和发展&#xff0c;计算机辅助设计&#xff08;CAD&#xff09;已经成为现代设计行业中不可或缺的一部分。在众多CAD软件中&#xff0c;Autodesk AutoCAD 2022&#xff08;CAD 2022&#xff09;无疑是最受欢迎和广泛应用的一款软件。作为一款全球领先的CAD软…

打造多平台游戏,Pygame让梦想照进现实

Pygame是一个流行的Python库&#xff0c;用于开发2D游戏。尽管它主要用于桌面游戏&#xff0c;但通过一些额外的工具和技巧&#xff0c;你也可以使用Pygame来打造多平台游戏&#xff0c;比如支持Windows、Linux、macOS、Android和iOS等平台。 下面是一个简单的Pygame游戏示例&…

sql 行列互换

在SQL中进行行列互换可以使用PIVOT函数。下面是一个示例查询及其对应的结果&#xff1a; 创建测试表格 CREATE TABLE test_table (id INT PRIMARY KEY,name VARCHAR(50),category VARCHAR(50) );向测试表格插入数据 INSERT INTO test_table VALUES (1, A, Category A); INSE…

某电力铁塔安全监测预警系统案例分享

项目概述 电力铁塔是承载电力供应的重要设施&#xff0c;它的安全性需要得到可靠的保障。但是铁塔一般安装在户外&#xff0c;分布广泛&#xff0c;且有很多安装在偏远地区&#xff0c;容易受到自然、人力的影响和破环。因此需要使用辅助的方法实时监控铁塔的安全状态&#xff…

ABC342 题解

ABC342 题解 A Description 给定一个串 s s s&#xff0c;求与 s s s 中其它字符不同的唯一字符的编号。 Solution 直接把 s s s 存到 t t t 里排序&#xff08;确实&#xff0c;sort(t.begin(), t.end()) 就可以&#xff09;&#xff0c;首尾特判再到 s s s 里查一下…

计算机设计大赛 深度学习大数据物流平台 python

文章目录 0 前言1 课题背景2 物流大数据平台的架构与设计3 智能车货匹配推荐算法的实现**1\. 问题陈述****2\. 算法模型**3\. 模型构建总览 **4 司机标签体系的搭建及算法****1\. 冷启动**2\. LSTM多标签模型算法 5 货运价格预测6 总结7 部分核心代码8 最后 0 前言 &#x1f5…

office word保存pdf高质量设置

1 采用第三方pdf功能生成 分辨率越大质量越好

MySQL集群 双主架构(配置命令)

CSDN 成就一亿技术人&#xff01; 今天刚开学第一天给大家分享一期&#xff1a;MySQL集群双主的配置需求和命令 CSDN 成就一亿技术人&#xff01; 神秘泣男子主页&#xff1a;作者首页 <———— MySQL专栏 &#xff1a;MySQL数据库专栏<———— MySQL双主是一…

学习笔记丨Shell

Usage of shell script References: Learn Shell, Shell 教程 | 菜鸟教程 The first line of shell script file begins with #!, followed by the full path where the shell interpreter is located. For example, #!/bin/bashTo find out the currently active shell and i…

python语言常见面试题:描述Python中的字典(Dictionary)和集合(Set)之间的区别。

Python中的字典&#xff08;Dictionary&#xff09;和集合&#xff08;Set&#xff09;是两种非常有用的数据结构&#xff0c;它们之间有一些明显的区别。 字典&#xff08;Dictionary&#xff09; 字典是一种无序的键值对集合。在字典中&#xff0c;每个键&#xff08;key&a…

Oracle 基础表管理(Heap-Organized Table Management)

表是数据库中负责数据存储的对象&#xff0c;在RDBMS中&#xff0c;数据以行、列的形式存储在表中。Oracle中表有很多种类型&#xff0c;最基础且应用最常用的类型就是堆表&#xff08;Heap-Organized Table&#xff09;&#xff0c;本文列举了Oracle堆表的常用管理操作。 一、…

Cpython和Jpython区别

Cpython和Jpython是Python语言的两种不同实现方式&#xff0c;它们之间存在一些关键的区别。 实现语言&#xff1a;Cpython是用C语言实现的&#xff0c;而Jpython则是用Java语言实现的。这意味着Cpython的源代码是用C语言编写的&#xff0c;而Jpython的源代码是用Java语言编写的…

pytorch --反向传播和优化器

1. 反向传播 计算当前张量的梯度 Tensor.backward(gradientNone, retain_graphNone, create_graphFalse, inputsNone)计算当前张量相对于图中叶子节点的梯度。 使用反向传播&#xff0c;每个节点的梯度&#xff0c;根据梯度进行参数优化&#xff0c;最后使得损失最小化 代码…

React Hooks概述及常用的React Hooks介绍

Hook可以让你在不编写class的情况下使用state以及其他React特性 useState ● useState就是一个Hook ● 通过在函数组件里调用它来给组件添加一些内部state,React会在重复渲染时保留这个state 纯函数组件没有状态&#xff0c;useState()用于设置和使用组件的状态属性。语法如下…

Qt的QThread、QRunnable和QThreadPool的使用

1.相关描述 随机生产1000个数字&#xff0c;然后进行冒泡排序与快速排序。随机生成类继承QThread类、冒泡排序使用moveToThread方法添加到一个线程中、快速排序类继承QRunnable类&#xff0c;添加到线程池中进行排序。 2.相关界面 3.相关代码 widget.cpp #include "widget…

实验室储样瓶耐强酸强碱PFA材质试剂瓶适用新材料半导体

PFA&#xff0c;全名可溶性聚四氟乙烯&#xff0c;试剂瓶又叫取样瓶、样品瓶、广口瓶、储样瓶等。主要用于痕量分析、同位素分析等实验室&#xff0c;广泛应用于新兴的半导体、新材料、多晶硅、硅材、微电子等行业。 规格参考&#xff1a;30ml、60ml、100ml、125ml、250ml、30…