动规解决01背包/完全背包精讲

还不会用动态规划解决01背包/完全背包?看这一篇文章就够了!

首先我们要明白什么是01背包和完全背包。

背包问题总体问法就是:

你有一个背包,最多能容纳的体积是V。

现在有n个物品,第i个物品的体积为vi​ ,价值为wi​。

现在有n种物品,每种物品有任意多个,第i种物品的体积为vi​ ,价值为wi​。

(1)求这个背包至多能装多大价值的物品?

(2)若背包恰好装满,求至多能装多大价值的物品?

根据物品个数是唯一还是无限多个,如果只能装一个,就是01背包问题;如果同一个物品能装无限多个,就是完全背包问题。

问法的区别:

同样也是分为两类,第一种就是必须将背包恰好装满,第二种问法是背包不必装满。

在弄清楚什么是01背包和完全背包后,我们来正式学习如何解决这类问题吧!


我们首先来详细讲解01背包和完全背包的母题(模板),然后会有相应的例题,同样也有详解给到大家!

一、01背包

【模板】01背包_牛客题霸_牛客网 (nowcoder.com)

我们先解决第一问,背包没有必须装满的情况下。

用动态规划解决问题有下面的标准步骤:

1、状态表示:

dp[i][j] 表示从前i个物品中挑选,总体积不超过j,所有选法中能挑选出的最大价值。

有同学会问,状态表示为什么是这样的呢?因为这样我们会包含物品个数和体积,也没必要多想,可以直接记住!

2、状态转移方程

根据最后一步的状况,分情况讨论。

这里的dp[i][j]分为两种情况:

  1. 不选i物品:dp[i-1][j]:此时状态表示就是第i-1个物品状态,直接照抄即可
  2. 选i物品:dp[i-1][j-v[i]]+w[i]:因为我们已经挑选了第i个物品,因此第i个物品的价值一定是先加上的。在我们选了第i个物品的情况下,我们就需要找在前i-1个物品中体积等于j-v[i]的状态。当然这里的前提是必须 j-v[i] 要大于等于0,从坐标要大于等于0 也可以看出!

综上,状态转移方程就是求这两者的最大值。

3、初始化

根据经验,我们必须多一层空间,防止下标越界。

根据转移方程,我们我们发现对于列是不会产生越界的,因为我们对于列下标都会有j-v[i] >= 0判断!

所以我们只需要考虑行初始化,下面的背包问题也是如此,列的下标越界问题不用考虑!只需要考虑行的初始化。

在第0行,表示在前0个物品中,总体积为j所表示的总价值,不存在,所以可以直接初始化为0。

4、填表顺序

根据状态转移方程可知,由上到下,由左到右。

5、返回值

由题意中的求这个背包至多能装多大价值的物品,所以我们返回dp[n][V].

n表示一共有n个物品,V表示背包所能容纳的最大体积。


我们继续解决第二问,背包必须装满的情况下。

1、状态表示

dp[i][j] 此时状态表示要与第一问进行区分:

dp[i][j] 表示从前i个物品中挑选,总体积正好等于j,所有选法中能挑选出的最大价值

2、状态转移方程

大部分内容与第一问相同,但是我们要考虑在前i个物品中挑选,可能体积要求不满足,也就是条件不存在的情况!

因此我们需要将不成立的部分要特殊处理!!!目的都是为了不要使用这些不存在的值

第一种,将不存在的情况赋值成-1.

第二种,将这些值赋值成0x3f3f3f3f,表示最大值,或者负的,表示最小值。

3、初始化
还是跟之前一样,第一列不需要初始化。

第一行的第一个数存在,赋值为0。但是后面的值就不存在,在前0个物品中,挑选出体积正好为1、2、3……这些情况都不存在,所以赋值为-1

4、填表顺序

从上往下

5、返回值

dp[n][V]


空间优化:

1、利用滚动数组在空间上的优化

我们可以直接用一维dp数组去代替二维数组

2、直接在原始的代码上稍加修改即可

直接将横坐标删除,然后遍历顺序修改成从右往左

为何遍历顺序改成从右往左?因为我们在初始化dp表时,用到了左上角的值,而一维滚动初始化时从左往右会导致新一轮的值会被覆盖、修改掉。因此需要从右往左进行初始化dp表!

空间优化后的代码

#include <iostream>
#include<bits/stdc++.h>
using namespace std;
 
int dp[1010];
int v[1010];
int w[1010];
 
int main()
{
    int n = 0, V = 0;
    cin >> n >> V;
 
    for(int i = 1; i <= n; i++)
    {
        cin >> v[i] >> w[i];
    }
 
    //解决第一问
    for(int i = 1; i <= n; i++)
    {
        for(int j = V; j >= v[i]; j--)
        {
            dp[j] = max(dp[j], w[i]+dp[j-v[i]]);
        }
    }
    cout << dp[V] << endl;
 
    //解决第二问
 
    memset(dp,0,sizeof(dp));
 
    for(int i = 1; i <= V; i++) dp[i] = -1;
 
    for(int i = 1; i <= n; i++)
    {
        for(int j = V; j >= 1; j--)
        {
            if(j >= v[i] && dp[j-v[i]] != -1)
            {
                dp[j] = max(dp[j], w[i]+dp[j-v[i]]);
            }
        }
    }
 
    if(dp[V] == -1)  cout << 0;
    else    cout << dp[V];
 
    return 0;  
}

二、完全背包

【模板】完全背包_牛客题霸_牛客网 (nowcoder.com)

我们先解决第一问,背包没有装满的情况下。

1、状态表示:

跟01背包状态表示一致。

dp[i][j] 表示从前i个物品中挑选,总体积不超过j,所有选法中能挑选出的最大价值。

2、状态转移方程

01背包和完全背包的本质区别就是能选择数量不一样,01背包数量只有1个,而完全背包可选择物品数量有无限多个

因此状态转移方程根据可选择物品数量分为很多种。

那有无限多种,如何将其转化为只有一种状态或者两种状态呢?

根据数学知识将纵坐标j 进行代换变成 j-v[i],进行如下证明即可得:

最终的方程就是:

dp[j] = max(dp[j], dp[j-v[i]] + w[i]);

这里可以直接记忆最后的答案,证明过程了解。

简记就是将第一个状态转移表达式中的横坐标加1即可!

3、初始化

只需要初始化第一行初始化为0即可

4、填表顺序

根据状态转移方程,从上往下填写每一行,每一行从左往右

5、返回值

dp[n][V]


我们继续解决第二问,背包必须装满的情况下。

1、状态表示

dp[i][j] 此时状态表示要与第一问进行区分:

dp[i][j] 表示从前i个物品中挑选,总体积正好等于j,所有选法中能挑选出的最大价值

2、状态转移方程

与第一问的区别就是:需要用-1额外表示不存在的状态。

3、初始化

将不存在的情况赋值为-1。

第一行除了第一个位置其余都不存在。

4、填表顺序

同第一问

5、返回值

同第一问

空间优化:

同样也是利用滚动数组进行空间优化。

注意这里与01背包的区别就是从左往右遍历。

区分:

01背包从右往左遍历的原因是他运用到了上一行的值,因为是横坐标是 i-1

而完全背包的状态转移方程的横坐标是

这两者状态转移方程的区别也决定了他们在初始化方向的问题!!!

虚线表示01背包的方向,实现表示完全背包的方向。

最终优化后的代码:

#include <iostream>
#include<bits/stdc++.h>
using namespace std;

int v[1010];
int w[1010];

int dp[1010];

int main()
{  
    int n = 0, V = 0;
    cin >> n >> V;
    for(int i = 1; i <= n; i++) cin >> v[i] >> w[i];

    //vector<vector<int>> dp(n+1, vector<int>(V+1));

    //解决第一问
    for(int i = 1; i <= n ; i++)
    {
        //从左往右遍历
        for(int j = 1; j <= V ; j++)
        {
            if(j-v[i] >= 0)
            {
                dp[j] = max(dp[j], dp[j-v[i]] + w[i]);
            }
        }
    }
    cout << dp[V] << endl;

    //解决第二问
    memset(dp,0,sizeof(dp));

    //先初始化为-1
    for(int i = 1; i <= V; i++) dp[i] = -1;

    for(int i = 1; i <= n ; i++)
    {
        for(int j = 1; j <= V ; j++)
        {
            if(j-v[i] >= 0 && dp[j-v[i]] != -1)
            {
                dp[j] = max(dp[j], dp[j-v[i]] + w[i]);
            }
        }
    }

    if(dp[V] == -1) cout << 0;
    else cout << dp[V];

    return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Linux|如何允许 awk 使用 Shell 变量

引言 当我们编写 shell 脚本时&#xff0c;我们通常会在脚本中包含其他较小的程序或命令&#xff0c;例如 awk 操作。就 Awk 而言&#xff0c;我们必须找到将一些值从 shell 传递到 Awk 操作的方法。 这可以通过在 Awk 命令中使用 shell 变量来完成&#xff0c;在本文中&#x…

做全域运营赛道,如何避免被割韭菜?

当下&#xff0c;全域运营赛道逐渐成型&#xff0c;许多创业者虽然都有了做全域运营服务商的想法&#xff0c;但却因全域运营是割韭菜等流言而心存疑虑&#xff0c;担心自己上当受骗&#xff0c;赔得血本无归。 事实上&#xff0c;关于全域运营是不是割韭菜这个问题&#xff0c…

Electron自动化测试技术选型调研

Electron简介 Electron是一个开源的框架&#xff0c;用于构建跨平台的桌面应用程序。它由GitHub开发并于2013年首次发布。Electron允许开发人员使用Web技术&#xff08;如HTML、CSS和JavaScript&#xff09;来构建桌面应用程序&#xff0c;同时可以在Windows、macOS和Linux等操…

量子计算机接入欧洲最快超算!芬兰加快混合架构算法开发

内容来源&#xff1a;量子前哨&#xff08;ID&#xff1a;Qforepost&#xff09; 文丨浪味仙 排版丨沛贤 深度好文&#xff1a;1900字丨7分钟阅读 摘要&#xff1a;芬兰技术研究中心&#xff08;VTT&#xff09;与 CSC 展开合作&#xff0c;基于量子计算机超算架构进行算法开…

Java为什么会成为现在主流的编程语言

Java为什么会成为现在的主流语言 前言一、Java语言概述Java是什么为什么大多数人会选择从事Java为什么从事Java的工作者数量从年递减 二、Java语言的特点简单性面向对象分布式&#xff08;微服务&#xff09;健壮性安全性体系结构中立可移植性解释型高性能多线程动态性 三、Jav…

Linux连接文件那点事

什么是连接文件 将一个文件和另一个文件建立联系&#xff0c;分为硬链接和软连接&#xff08;符号连接&#xff09;。 硬链接 Linux中&#xff0c;所有的文件都有一个inode&#xff0c;这个东西就是文件的ID号&#xff0c;硬链接的方式就是通过这个inode来产生新的文件名来建…

ChatGPT4O:自然语言交互

ChatGPT 4O&#xff1a;引领自然语言处理的新纪元 一、技术细节与强大功能二、创新点与技术突破三、应用场景与商业化前景 在科技的浪潮中&#xff0c;自然语言处理&#xff08;NLP&#xff09;领域一直备受关注。最近&#xff0c;OpenAI公司发布了其最新的NLP模型——ChatGPT …

echarts 环形图实现透明间隔,嵌套环形图片和图形

echarts 环形图实现透明间隔&#xff0c;嵌套环形图片和图形 环形图实现透明间隔环形图嵌套环形图片环形图嵌套环形图形 环形图实现透明间隔 首先通过 radius 属性实现一个圆环图 再通过 padAngle 属性设置扇区角度即可 使用 borderRadius 属性设置扇形区块的内外圆角半径&…

【PDF技巧】PDF如何解密?

PDF文件设置了加密&#xff0c;需要密码才能够打开文件或者编辑文件&#xff0c;那么如何解密PDF密码&#xff1f;今天我们来一起学习一下。 首先是在已知密码的情况下&#xff0c;PDF文件中的打开密码或者是限制编辑&#xff0c;想要解密PDF密码&#xff0c;我们只需要在PDF编…

这些网站我愿称之为做答辩PPT的神

很多快要毕业的同学在做答辩PPT的时候总是感觉毫无思路&#xff0c;一窍不通。但这并不是你们的错&#xff0c;对于平时没接触过相关方面&#xff0c;第一次搞答辩PPT的人来说&#xff0c;这是很正常的一件事。一个好的答辩PPT可以根据以下分为以下几部分来写。 1.研究的背景和…

寒冷地区污水处理一体化设备如何选型

在寒冷地区选择污水处理一体化设备时&#xff0c;需要特别考虑设备的耐寒性能和处理效率&#xff0c;以确保在低温环境下仍能稳定运行并达到预期的处理效果。以下是一些选型时需要考虑的关键因素&#xff1a; 耐寒设计&#xff1a;选择具有耐寒设计的设备&#xff0c;例如&…

【GESP】2023年12月图形化一级 -- 返回地球

返回地球目 1. 准备工作 (1)删除默认小猫角色。 (2)添加角色Cat Flying和Earth。 (3)删除默认白色背景,添加背景Nebula。 2. 功能实现(根据注意事项,用积木块实现下列描述的功能) (1)点击绿旗,角色Earth的初始位置为(X=170,Y=120),并以每秒向右旋转15度的…

刘邦痛恨的叛徒雍齿,为何后来还被封了侯?

雍齿&#xff0c;原是沛县的世族出身&#xff0c;家庭往上追溯几代&#xff0c;也曾经显赫过。 虽然比不上先祖世代为楚将的项梁、项羽&#xff0c;但雍齿这个没落的世族后代&#xff0c;身上多多少少也还讲究点贵族遗风。 战国时期&#xff0c;以秦国的军功爵制为代表&#…

【Uniapp】简易封装提示框showToast/showModal

组件 // 封装提示框 export const showModal (content, showCancel false, title "提示") > {return new Promise((resolve, reject) > {uni.showModal({title: title,content: content,showCancel: showCancel,success: (res) > {resolve(res);},fail:…

CTF网络安全大赛web题目:just_sqli

这道题目是bugku的web题目 题目的 描  述: KosenCTF{} 原文链接&#xff1a; CTF网络安全大赛web题目&#xff1a;just_sqli - 红客网-网络安全与渗透技术 题目Web源代码&#xff1a; <?php$user NULL; $is_admin 0;if (isset($_GET["source"])) {highlig…

每日学习 - APK解包

文章目录 APK的定义解析APKAPK 是什么每个文件的意义classes.dexAndroidManifest.xmlassetslibres & resources.arsc 反编译工具apktool apk解包 秒了~ APK的定义 APK&#xff08;Android Package Kit&#xff09;是用于部署和分发Android操作系统上应用程序的软件包格式。…

谷粒商城实战(024 业务-订单模块-分布式事务1)

Java项目《谷粒商城》架构师级Java项目实战&#xff0c;对标阿里P6-P7&#xff0c;全网最强 总时长 104:45:00 共408P 此文章包含第284p-第p290的内容 简介 模拟积分服务出异常&#xff0c;前方的锁库存事务未回滚&#xff0c;这时候就需要分布式事务 本地事务 事务的隔离…

Linux环境部署与命令技巧

Linux环境部署与命令技巧 安装Java 首先确保已经安装了Java。通过运行以下命令检查Java是否已安装&#xff1a; java -version # 查看Java版本如果未安装Java&#xff0c;可以使用以下命令安装OpenJDK&#xff1a; sudo yum install java-11-openjdk # 安装OpenJDK 11创建一个…

华为昇腾310B1平台视频解码失败[ERROR] Send frame to vdec failed, errorno:507018

目录 1 [ERROR] Send frame to vdec failed, errorno:507018 2 bug解决尝试1 3 bug解决尝试2 4 最终解决方法 参考文献&#xff1a; 1 [ERROR] Send frame to vdec failed, errorno:507018 某项目中的代码运行报错 [ERROR] Send frame to vdec failed, errorno:507018 Ac…