计算机算法分析与设计(9)---0-1背包和完全背包问题(含C++代码)

news2025/2/1 8:47:59

文章目录

  • 一、0-1背包概述
    • 1.1 问题描述
    • 1.2 算法思想
  • 二、0-1背包代码
    • 2.1 题目描述
    • 2.2 代码编写
  • 三、完全背包概述
  • 四、完全背包代码
    • 4.1 题目描述
    • 4.1 代码编写
    • 4.2 代码优化


一、0-1背包概述

1.1 问题描述

 1. 0-1背包问题:给定 n n n 种物品和一背包。物品 i i i 的体积是 v i v_i vi,其价值为 w i w_i wi,背包的容量为 c c c。问应如何选择装入背包的物品,使得装入背包中物品的总价值最大?

 2. 在选择装入背包的物品时,对每种物品 i i i 只有两种选择,即装入背包和不装入背包。不能将物品 i i i 装入背包多次,也不能只装入部分的物品 i i i

1.2 算法思想

 1. f [ i ] [ j ] f[i][j] f[i][j] 定义:前 i i i 个物品,背包容量 j j j 下的最优解(最大价值)。

  • 当前的状态依赖于之前的状态,可以理解为从初始状态 f [ 0 ] [ 0 ] = 0 f[0][0] = 0 f[0][0]=0 开始决策,有 n n n 件物品,则需要 n n n 次决策。每一次对第 i i i 件物品的决策,状态 f [ i ] [ j ] f[i][j] f[i][j] 不断由之前的状态更新而来。

 2. 当前背包容量不够( j < v [ i ] j < v[i] j<v[i]),没得选,因此前 i i i 个物品最优解即为前 i − 1 i−1 i1 个物品最优解。

  • 对应代码:f[i][j] = f[i - 1][j]

 3. 当前背包容量够( j > v [ i ] j > v[i] jv[i]),可以选,因此需要决策选与不选第 i i i 个物品。

  • 选:f[i][j] = f[i - 1][j - v[i]] + w[i]
  • 不选:f[i][j] = f[i - 1][j]
  • 我们的决策是如何取到最大价值,因此以上两种情况取 max() 。

二、0-1背包代码

2.1 题目描述

在这里插入图片描述

输入样例:
4 5
1 2
2 4
3 4
4 5
输出样例:
8

2.2 代码编写

在这里插入图片描述

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

const int MAXN = 1005;
int v[MAXN];    // 体积
int w[MAXN];    // 价值 
int f[MAXN][MAXN];  // f[i][j], j体积下前i个物品的最大价值 

int main() 
{
    int n, m; //n表示物品的数量,m表示背包容积 
    cin >> n >> m;
    for(int i = 1; i <= n; i++) 
        cin >> v[i] >> w[i]; //从1开始输入每个物品的体积和价值
        
    for(int i = 0; i <= m; i++) //设置在没有物品情况下,无论背包有多少体积,价值都为0
        f[0][i]=0;
    for(int j = 0; j <= n; j++) //设置在背包体积为0情况下,无论有多少物品,价值都为0
        f[j][0]=0;
        
    for(int i = 1; i <= n; i++) 
        for(int j = 1; j <= m; j++)
        {
            //  当前背包容量装不进第i个物品,则价值等于前i-1个物品
            if(j < v[i]) 
                f[i][j] = f[i - 1][j];
            // 能装,需进行决策是否选择第i个物品
            else    
                f[i][j] = max(f[i - 1][j], f[i - 1][j - v[i]] + w[i]);
        }           

    cout << f[n][m] << endl;

    return 0;
}

三、完全背包概述

 1. 思路同0-1背包问题。区别在于0-1背包对于每种物品只有选或不选,这也0-1的由来。而完全背包则对于每种物品可以多次选择。

 2. 因为选择物品的总体积不能超过 j j j ,所以第 i i i 件物品最多选 j / v i j / v_i j/vi(向下取整件)。

 3. f [ i ] [ j ] f[i][j] f[i][j] 定义:前 i i i 个物品,背包容量 j j j 下的最优解(最大价值)。
   f [ i ] [ j ] = m a x ( f [ i − 1 ] [ j ] , f [ i − 1 ] [ j − v i ] + w i , f [ i − 1 ] [ j − 2 ∗ v i ] + 2 ∗ w i , f [ i − 1 ] [ j − k ∗ v i ] + k ∗ w i , . . . . . ) f[i] [j] = max( f[i-1][j] , f[i - 1][j - v_i]+w_i , f[i - 1][j - 2 * v_i] + 2 * w_i , f[i - 1][j - k * v_i ] + k * w_i , .....) f[i][j]=max(f[i1][j],f[i1][jvi]+wi,f[i1][j2vi]+2wi,f[i1][jkvi]+kwi,.....)

四、完全背包代码

4.1 题目描述

在这里插入图片描述

输入样例:
4 5
1 2
2 4
3 4
4 5
输出样例:
10

4.1 代码编写

#include<iostream>
using namespace std;

const int N = 1010;

int n, m;
int f[N][N]={0}, v[N], w[N];

int main(){
    cin >> n >> m;
    for(int i = 1; i <= n; i ++ )
        cin >> v[i] >> w[i];
        
    for(int i = 1; i <= n; i ++ )
        for(int j = 1; j <= m; j ++ )
            for(int k = 0; k * v[i] <= j; k ++ )
                f[i][j] = max(f[i][j], f[i - 1][j - k * v[i]] + k * w[i]);
                
    cout << f[n][m] << endl;
    return 0;
}

4.2 代码优化

 1. f [ i ] [ j ] = m a x ( f [ i − 1 ] [ j ] , f [ i − 1 ] [ j − v i ] + w i , f [ i − 1 ] [ j − 2 ∗ v i ] + 2 ∗ w i , f [ i − 1 ] [ j − k ∗ v i ] + k ∗ w i , . . . . . ) f[i] [j] = max( f[i-1][j] , f[i - 1][j - v_i]+w_i , f[i - 1][j - 2 * v_i] + 2 * w_i , f[i - 1][j - k * v_i ] + k * w_i , .....) f[i][j]=max(f[i1][j],f[i1][jvi]+wi,f[i1][j2vi]+2wi,f[i1][jkvi]+kwi,.....)

   f [ i ] [ j − v i ] = m a x ( f [ i − 1 ] [ j − v i ] , f [ i − 1 ] [ j − 2 ∗ v i ] + w i , f [ i − 1 ] [ j − 3 ∗ v i ] + 2 ∗ w i , . . . . . ) f[i][j - v_i] = max( f[i-1][j - v_i] , f[i-1][j - 2 * v_i] + w_i , f[i-1][j - 3 * v_i] + 2 * w_i , .....) f[i][jvi]=max(f[i1][jvi],f[i1][j2vi]+wi,f[i1][j3vi]+2wi,.....)

  由上面的两式,可得出如下递推关系:f[i][j] = max(f[i][j-v[i]] + w[i] , f[i-1][j])

 2. 去掉第三层的 k k k 循环,优化后的代码如下:

#include<iostream>
using namespace std;

const int N = 1010;

int n, m;
int f[N][N]={0}, v[N], w[N];

int main()
{
    cin >> n >> m;
    for(int i = 1; i <= n; i ++ )
        cin >> v[i] >> w[i];
    for(int i = 1; i <= n; i ++ )
    {
        for(int j = 1; j <= m; j ++ )
        {
            if(v[i] <= j)
                f[i][j] =max(f[i - 1][j], f[i][j - v[i]] + w[i]);
            else
                f[i][j] = f[i - 1][j];
        }
    }
    cout << f[n][m] << endl;
    return 0;
}

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

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

相关文章

Unity可视化Shader工具ASE介绍——7、ASE实现Matcap效果和自定义节点

大家好&#xff0c;我是阿赵。继续介绍Unity可视化Shader编辑工具ASE。上一篇用了很长的篇幅来做了一个遮挡X光的效果。这一篇来做一个MatCap效果。不过做MatCap并不是目的&#xff0c;是想说明一下&#xff0c;怎样在ASE里面自定义方法节点。 一、在ASE里面做MatCap材质 由于…

【每日一题Day348】LC137只出现一次的数字Ⅱ | 状态转移

只出现一次的数字Ⅱ【LC137】 给你一个整数数组 nums &#xff0c;除某个元素仅出现 一次 外&#xff0c;其余每个元素都恰出现 **三次 。**请你找出并返回那个只出现了一次的元素。 你必须设计并实现线性时间复杂度的算法且不使用额外空间来解决此问题。 排序 将数组从小到大…

LED灯的基本控制

题目要求 首先让8路LED指示灯闪烁3遍然后熄灭&#xff0c;接着依次点亮LED指示灯&#xff0c;最后依次熄灭LED灯&#xff0c;程序循环实现上述功能。 分析设计 原理图&#xff1a; 首先应该选通Y4C&#xff0c;也就是ABC 100。 代码 #include <regx52.h> #define…

C语言 —— 指针

目录 1. 指针是什么&#xff1f; 2. 指针和指针类型的关系 2.1 指针的解引用 2.2 指针-整数 3. 野指针 3.1 野指针成因 1. 指针未初始化 2. 指针越界访问 3. 指针指向的空间释放 3.2 如何规避野指针 4. 指针运算 4.1 指针-整数 4.2 指针-指针 指针-指针的使用 4.3 指针的关系运…

3.7 static关键字

思维导图&#xff1a; 3.7.1 静态属性 ### 3.7 static关键字 --- Java提供了一个称为static的关键字&#xff0c;用于修饰类的成员&#xff0c;如成员变量、成员方法和代码块。使用static修饰的成员拥有特殊性。 --- #### 3.7.1 静态属性 当属性被static关键字修饰时&…

嵌入式实时操作系统的设计与开发(轮询系统学习)

轮询系统具有以下工作特点&#xff1a;系统完成一个轮询的时间取决于循环中需要执行的函数个数。此外&#xff0c;轮询的次序是静态固定的&#xff0c;在运行时不能进行动态调整。 典型系统 许多工业线程网络中&#xff0c;由于需要控制的设备较多、相互距离又较远&#xff0…

手把手教你用Python绘制神经网络图

接下来教大家如何使用 Python 中的 networkx 库&#xff0c;绘制美观且标准的神经网络。会根据指定的层和节点数量&#xff0c;绘制不同结构的神经网络。 networkx 库可以用来创建和操作图类型的数据结构&#xff0c;其中包括无向图、有向图、带权图等等。 神经网络可以看做是一…

字节码之 Lambda 表达式底层原理

文章目录 0.前言0. lambda程序示例1. 编译程序&#xff1a;2. 使用 javap 分析字节码3. 输出字节码4. 分析指令 1. Lambda 表达式的字节码实现1.1 什么是invokedynamic 指令invokedynamic 的工作原理为何 invokedynamic 如此特殊&#xff1f; 1.2 bootstrap method 详解1.1 Lam…

Qt之给控件添加右键菜单

一、设置控件 在对应控件的属性中&#xff0c;将contextMenuPolicy设置为CustomContextMenu。 二、添加槽函数 在对应控件上右键选择槽函数customContextMenuRequested(QPoint)。 三、在槽函数中添加右键菜单 在槽函数中输入如下代码&#xff0c;添加右键菜单。 //右键菜单 …

红帽Linux的安装和部署

目录 一、红帽Linux的安装阶段 1、下载redhat7.9的iso镜像 2、安装阶段 二、红帽Linux的配置阶段 1、第一次进入装机配置 2、进入机器后的一些配置 三、远程连接阶段 1、关闭防火墙 2、使用Xshell远程连接&#xff08;其他连接工具也行&#xff09; 1.开启SSH服务 2.连…

二十、【钢笔工具组】

文章目录 钢笔工具自由钢笔工具弯度钢笔工具 钢笔工具 钢笔工具在photoshop作图中是一款使用频率较高的路径工具,我们可以在窗口选项栏中将路径编辑栏打开&#xff0c;如果我们需要选中使用路径&#xff0c;需要用到后边的路径工具才能去拖动&#xff0c;而选择工具不能拖动&a…

9月大型语言模型研究论文总结

大型语言模型(llm)在今年发展迅速&#xff0c;随着新一代模型不断地被开发&#xff0c;研究人员和工程师了解最新进展变得非常重要。本文总结9-10月期间发布了一些重要的LLM论文。 这些论文涵盖了一系列语言模型的主题&#xff0c;从模型优化和缩放到推理、基准测试和增强性能…

Sigma中的数字增益放大/降低方法

1 是否需要申请加入数字音频系统研究开发交流答疑群(课题组)&#xff1f;加他微信hezkz17, 本群提供音频技术答疑服务

如何快速分析一款产品?

一、何时需要对一个产品进行分析&#xff1f; 首先&#xff0c;当你刚刚融入一个新的产品团队&#xff0c;尤其是当你需要深入了解你将负责的产品时&#xff0c;分析产品就显得尤为重要。这有助于你对产品的全面理解&#xff0c;发现其中的优势和不足&#xff0c;为未来的工作提…

14.5 Socket 应用组播通信

组播通信是一种基于UDP协议的网络通信方式&#xff0c;它允许发送方将消息同时传递给多个接收方。在组播通信中&#xff0c;发送方和接收方都会加入一个共同的组播组&#xff0c;这个组播组对应一个特定的IP地址&#xff0c;所有加入该组播组的主机都能够接收到发送方发送的消息…

C++概述

一、C特色 1.C是面向对象的高级程序设计语言 2.支持数据封装&#xff0c;将数据和对该数据进行操作的函数封装在一个类中&#xff0c;对象就是某一个具体的类。即类是数据封装的工具&#xff0c;对象是数据封装的实现。 3.具有继承性 4.具有函数重载 二、拓展介绍 1.C标准&a…

基于音频SOC开发板的主动降噪ANC算法源码实现

基于音频SOC开发板的主动降噪ANC算法源码实现 是否需要申请加入数字音频系统研究开发交流答疑群(课题组)?可加我微信hezkz17, 本群提供音频技术答疑服务,+群附加赠送降噪开发资料,

IO流:java中解码和编码出现乱码说明及代码实现

IO流&#xff1a;java中解码和编码的代码实现 一、UTF-8和GBK编码方式二、idea和eclipse的默认编码方式三、解码和编码方法四、代码实现编码解码 五、额外知识扩展 一、UTF-8和GBK编码方式 如果采用的是UTF-8的编码方式&#xff0c;那么1个英文字母 占 1个字节&#xff0c;1个…

深度学习-优化算法与梯度下降

文章目录 前置知识指数移动平均EMAL2正则&#xff08;L2 Regularization)权重衰减&#xff08;Weight Decay) 优化器SDGMomentumNAGAdagradRMSPropAdaDeltaAdamAdamW综上 学习率StepLRMultiStepLRExponentialCosineAnealingRLROP&#xff08;ReduceLRonPlateau)lambda总结 前置…

【图像处理】图像配准、图像增强和图像分割研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…