【20221206】【每日一题】01背包的基础

news2024/9/23 11:28:32


思路:

二维数组

动规五部曲

1、确定dp数组以及下标含义:二维数组dp[i][j]表示从下标为0-i的物品里任意取,放入容量为j的背包,价值总和最大为多少
2、确定递推关系式:从两个方向推dp[i][j],没放物品i以及放了物品i。没放物品i,由dp[i-1][j](当物品i的重量大于背包j的重量时,物品i无法放进背包中,所以被背包内的价值依然和前面相同),放了物品i,由dp[i-1][j-weight[i]]+value[i],也就模拟了物品i放入后得到的价值。因此递推关系式为:dp[i][j]=max(dp[i-1][j],dp[i-1][j-weight[i]]+value[i])
3、初始化:背包容量为0时,一定为0;由递推关系可知i状态的价值由i-1状态推导而来,所以物品0的状态都需要初始化,即dp[0][j]。
4、确定遍历顺序:有两种遍历维度,物品与背包重量,先遍历物品更好理解。
5、举例推导dp数组。

void test() {
    vector<int> weight = { 1,3,4,1 };
    vector<int> value = { 15,20,40,25};
    int bagweight = 4;
    //初始化
    vector<vector<int>> dp(weight.size(), vector<int>(bagweight + 1, 0));
    for (int i = weight[0]; i <= bagweight; i++) 
    {
        dp[0][i] = value[0];
    }
    //两个维度的遍历,先遍历物品,再遍历背包
    for (int i = 1; i < weight.size(); i++)
    {
        for (int j = 1; j <= bagweight; j++)
        {
            if (j < weight[i]) dp[i][j] = dp[i - 1][j];//物品i没放进来
            else  dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);
        }
    }
    cout << dp[weight.size() - 1][bagweight] << endl;
}
int main()
{
    test();
}

滚动数组

如果将dp[i-1]那层的数据都拷贝到dp[i]那一层,递推公式其实可以变为:dp[i][j]=max(dp[i][j],dp[i][j-weight[i]]+value[i])。
只用一个一维数组来实现,即滚动数组。能用滚动数组的条件是:上一层可以重复利用,直接拷贝到当前层。

动规五部曲
1、dp数组的意义:dp[j]表示容量为j的背包最大价值为dp[j]
2、递推关系式:dp[j]可以通过dp[j-weight[i]]推导,dp[j]=max(dp[j],dp[j]-weight[i]+value[i])
3、初始化:如果价值都大于0的话,刚开始的时候都初始化为0即可。
4、确定遍历顺序:先物品再背包,注意这里遍历背包的时候是从大到小倒序的目的是为了保证物品i只被放入了一次,从后往前循环,每次取得状态不会和之前取得状态重合,这样每种物品就只取一次了。 注意终止条件:for(int j = bagWeight; j >= weight[i]; j--)
5、举例推导dp数组。

void test01() {
    vector<int> weight = { 1,3,4,1 };
    vector<int> value = { 15,20,40,25 };
    int bagweight = 4;
    
    //初始化
    vector<int> dp(bagweight + 1, 0);
    for (int i = 0; i < weight.size(); i++)
    {
        for (int j = bagweight; j >= weight[i]; j--)//背包倒序
        {
            dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);
        }
        for_each(dp.begin(), dp.end(), [=](int x) {cout << x << " "; });
        cout << endl;
    }
    //cout << dp[bagweight] << endl;

}
int main()
{
    test01();
}

将每一次都打印出来,其实发现和二维数组是一样的,节省了空间复杂度。 

 

 

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

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

相关文章

CSS 之 渐变色边框

一、渐变色边框 如果我们前端最亲爱的UI设计师&#xff0c;让我们给盒子绘制一个渐变色的边框&#xff0c;而且盒子的宽高还需要随着内容变化而变化&#xff0c;那我们就不能通过切图来解决问题&#xff0c;所以我们可以这么说&#xff1a; 但是我相信优秀的你肯定不会说做不…

如何删除密码?知道密码和不知道密码的情况

压缩包设置了密码&#xff0c;就需要输入压缩包密码才能顺利解压文件。 有些时候我们加密了压缩包之后&#xff0c;过了一段时间可能就不需要再加密压缩包里的文件了。 有些时候我们加密了压缩包之后&#xff0c;长时间没有使用&#xff0c;又没有将密码记录在一个地方&#…

基于 LSTM 的分布式能源发电预测(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️❤️&#x1f4a5;&#x1f4a5;&#x1f4a5; &#x1f389;作者研究&#xff1a;&#x1f3c5;&#x1f3c5;&#x1f3c5;主要研究方向是电力系统和智能算法、机器学…

CNAME记录和A记录

文章目录定义CNAME使用说明总结定义 A记录&#xff1a;A (Address) 记录是用来指定主机名&#xff08;或域名&#xff09;对应的IP地址记录。CNAME记录&#xff1a;CName记录是Canonical Name的简称&#xff0c;通常称别名指向&#xff0c;CNAME记录可用于将一个域名别名为另一…

太强了!GitHub大佬白嫖的SpringCloud微服务进阶宝典,啃完感觉能吊锤面试官!

自 2014 年起&#xff0c;微服务技术一直火热至今。随着越来越完善的微服务技术栈的发布&#xff0c;以及越来越多的微服务项目实际的落地和上线&#xff0c;使用 Java 技术栈的企业应该都在尝试或者已经落地了各自的微服务项目。同时&#xff0c;通过招聘网站的信息和每次面试…

如何用蓝牙实现无线定位(五)--双定位显示

1. 机器人定位装置的构建 按照上面的针脚使用杜邦线将救援机器人定位装置的主从蓝牙连接到主控板上&#xff0c;注意错误的连接会导致模块损坏【参考视频】 2. 机器人位置的获取与发送 (1) 在机器人定位装置的控制板中烧录以下程序&#xff08;robot.ino&#xff09; /*-----…

深度解读汽车域控制器

已剪辑自: https://mp.weixin.qq.com/s?__bizMzg4NjIxODk4Mg&mid2247496089&idx1&sndb5c08f97342decfabc9ce985ec75aaa&chksmcf9fb810f8e83106994f2f2b9ca0387eaca7543d36b1673d4fc00bdfe07fbc5099322d41a702&scene21#wechat_redirect * * 过去十多年的…

华为机试-字符串合并处理

描述 按照指定规则对输入的字符串进行处理。 详细描述 第一步&#xff1a;将输入的两个字符串str1和str2进行前后合并。如给定字符串 “dec” 和字符串 “fab” &#xff0c; 合并后生成的字符串为 “decfab” 第二步&#xff1a;对合并后的字符串进行排序&#xff0c;要求…

拖拽式网页制作工具

拖拽式网页制作工具是什么&#xff0c;有什么优势&#xff0c;怎么使用&#xff1f; 拖拽式网页制作工具是什么: 它是一款可以帮助企业、机构、个体户快速搭建网站的一款工具&#xff0c;通过简单易上手的操作&#xff0c;免除学习代码、学习设计等繁琐步骤&#xff0c;利用工…

2022.12.2Treats for the Cows POJ - 3186(区间dp

原题链接&#xff1a;传送门 FJ has purchased N (1 < N < 2000) yummy treats for the cows who get money for giving vast amounts of milk. FJ sells one treat per day and wants to maximize the money he receives over a given period time. The treats are inte…

请不要直接拆除或更换旧光纤!Softing为您提供光纤以太网网络解决方案

Softing的Phoenix Digital光纤以太网网络冗余模块与传统光纤相兼容。通过使用Phoenix Digital模块&#xff0c;用户无需更换传统光纤和远程I/O设备。 升级工业网络是一项复杂的工作&#xff0c;用户需要耗费大量的时间、成本和其他资源来确保新建网络可以满足系统应用中的所有要…

Excel - 插入空白行

简单的插入一个空白行&#xff0c;只需选中一行&#xff0c;右键&#xff0c;选择插入一行即可。 如果要一次插入多个空行&#xff0c;可以按住Ctrl键&#xff0c;然后逐个选中想要插入的行&#xff0c;然后执行插入操作&#xff0c;如下所示&#xff1a; 选中张三、王五、田七…

二叉树的基本运算

二叉树的基本运算 上一讲我们已经讲了创建二叉树,所以这一讲,我们来说一下二叉树的基本运算方法,为以后二叉树的运用打下基础: (1)查找节点FindNode(*b,x): 在二叉树b中寻找data域值为x的节点,并返回指向该节点的指针 (2)找孩子节点LchildNode和Rchild-Node: 分别求二叉树中节…

逆傅里叶变IFFT原始信号恢复方法研究-附Matlab代码

一、原始信号模拟 现实中&#xff0c;由于得到的信号都是实数序列。设有实数序列x(n)&#xff0c;如下图所示 图1 原始信号 对应Matlab如下&#xff1a; %% 矩形波 N33; % 设置N长 xzeros(1,N); % 构成矩形波形 x(7:27)1; figur…

JavaSe-JAVA的多态

用一句话概括就是&#xff1a;事物在运行过程中存在不同的状态。先以教科书般举例说明,下文再举一个花木兰替父从军的例子帮助大家理解. 多态的存在有三个前提: 1.要有继承关系 2.子类要重写父类的方法 3.父类引用指向子类对, 但是其中又有很多细节需要注意。首先我们定义两…

java代码审计

java代码审计 SQL注入分析 1、JDBC 1、1&#xff08;statement&#xff09; public String jdbcVul(String id) {StringBuilder result new StringBuilder();try {Class.forName("com.mysql.cj.jdbc.Driver");Connection conn DriverManager.getConnection(db_u…

Qt OpenGL(二十六)——Qt OpenGL 核心模式-旋转的彩色三角形

Qt OpenGL(二十六)——Qt OpenGL 核心模式-旋转的彩色三角形 上篇文章我们把三角形上色成了,彩色的三角形: 离我们想要绘制的图形,旋转的三角形,又近了一步,本篇文章就让这个彩色的三角形旋转起来。不过从上篇开始使用的代码就是Qt封装的类了,不过也是核心模式的,步…

【Qt】从开源项目QCAD中学习如何增强QLineEdit控件

1 背景 Qt5 中有个很基本的单行输入框控件&#xff0c;就是 QLineEdit&#xff0c;类似于HTML中的input标签。 在QCAD开源项目中&#xff0c;其主界面中有个供用户输入绘图命令的单行输入框控件&#xff0c;其可以实现类似 Linux 终端的简单效果&#xff1a; 上键显示最近的…

基于jsp+ssm的家政服务系统-计算机毕业设计

项目介绍 作为一个家政服务系统&#xff0c;它就面对着许多用户的操作&#xff0c;而这些用户对系统的操作应该有所不同&#xff0c;所以我们在设计时必须要对不同的用户设立不同的的权限在本系统中&#xff0c;我考虑了3种权限&#xff08;管理员&#xff0c;雇主&#xff0c…

盘点JDK中基于CAS实现的原子类

前言 JDK中提供了一系列的基于CAS实现的原子类&#xff0c;CAS 的全称是Compare-And-Swap&#xff0c;底层是lock cmpxchg指令&#xff0c;可以在单核和多核 CPU 下都能够保证比较交换的原子性。所以说&#xff0c;这些原子类都是线程安全的&#xff0c;而且是无锁并发&#x…