动态规划DP之背包问题3---多重背包问题

目录

DP分析:

优化:

 二进制优化

例题:


        01背包是每个物品只有一个,完全背包问题是每个物品有无限个。

        那么多重背包问题就是 每个物品有有限个

有 N 种物品和一个容量是 V 的背包。

第 i 种物品最多有 si 件,每件体积是 vi,价值是 wi。

求解将哪些物品装入背包,可使物品体积总和不超过背包容量,且价值总和最大。
输出最大价值。

DP分析:

        和完全背包问题很像,暴力算法都是多加一层循环,循环物品的个数。O(n^3)

动态规划DP之背包问题2---完全背包问题-CSDN博客

         实现代码:

for(int i=1;i<=n;i++){for(int j=1;j<=V;j++){f[i][j] = f[i-1][j];for(int k=0;k<=s[i]&&k*v[i]<=j;k++)f[i][j] = Math.max(f[i][j],f[i-1][j-k*v[i]]+k*w[i]);                    }
}

优化:

        不能采用完全背包的优化方式。动态规划DP之背包问题2---完全背包问题-CSDN博客 

        因为:

f[i,j]=max(f[i-1,j],f[i-1,j-v]+w,...\ f[i-1,j-sv]+sw) 

f[i,j-v]=max(\ f[i-1,j-v],f[i-1,j-2v]+w,...\ f[i-1,j-sv]\\+(s-1)w,\ f[i-1,j-(s+1)v]+sw)

        多了一个 f[i-1,j-(s+1)v]+sw,而max是不能减少一个获取到最大值的。

 二进制优化

        和快速幂的思路方法很像。快速幂(求解原理+例题)-CSDN博客

    假如:物品 i 的数量为 s_i=1023

  • 暴力做法就是从 1 枚举到 1023 。
  • 使用二进制优化,我们只需要枚举 (int)(log_2s_i)+1 个数:1,2,4,8,16,...512,就可以组合出 [0,1023] 中的任意一个整数。(相当于二进制表示转化为十进制) 

一般性下,如和求出 s_i\ log下需要的数是哪些:

   2^0,\ 2^1,\ 2^2,\ 2^3,\ ...2^k, \ c\ \ (c=s_i-2^0-2^1-...2^k,\ c<2^{k+1})

   从 2^0 到 2^k 可以组合成 [0,2^{k+1}-1] 之间的任何一个数,加上 c 后可以组成 [c,2^{k+1}-1+c],其中 2^{k+1}-1+c = s_i 。

   如何保证第一段 [0,2^{k+1}-1] 与 [c,2^{k+1}-1+c] 之间没有空隙,即 2^{k+1}-1 是否大于 c

        因为 c 的取值 c<2^{k+1} 保证了

         2^0+2^1+2^2+2^3+ ...2^k+ 2^{k+1} >s_i\\2^0+2^1+2^2+2^3+ ...2^k\ \ \ \ \ \ \ \ \ \leqslant s_i\\ 2^0+2^1+2^2+2^3+ ...2^k+ \ c \ \ \ =s_i

        如果 c>=2^{k+1},那么就会取 2^{k+1} ,不是 c 了。

因此,我们将 s_i\rightarrow logs_i,然后针对分开后的所有物品使用01背包处理方式。时间复杂度降为O(n^2logs)

优化代码:

        转换为01背包问题,将 s_i 拆后的所有数,分别作为一种物品的数量。

for(int i=1;i<=n;i++){str = in.readLine().split(" ");int vi = Integer.parseInt(str[0]); // 物品i的体积int wi = Integer.parseInt(str[1]); // 物品i的价值int si = Integer.parseInt(str[2]); // 物品i的数量// 直接将该物品数目拆分成多个,但是拆分完后的物品数目可以组合成si中的任何一个数目int k = 1; // 从1开始划分,每次乘23while(si>=k){ //满足c<2^(k+1) ,即是s大于k,才能划分k个物品出去v[cnt] = vi*k; // 个数*体积,作为新一个物品w[cnt] = wi*k;si -= k; // 减去划分的k *= 2; cnt++;}if(si!=0){ // 最后剩下的物品,即cv[cnt] = si*vi;w[cnt] = si*wi;cnt++;}
}
n = cnt;

例题:

有 N 种物品和一个容量是 V 的背包。

第 i 种物品最多有 si 件,每件体积是 vi,价值是 wi。

求解将哪些物品装入背包,可使物品体积总和不超过背包容量,且价值总和最大。
输出最大价值。

输入格式

第一行两个整数,N,V,用空格隔开,分别表示物品种数和背包容积。

接下来有 N 行,每行三个整数 vi,wi,si,用空格隔开,分别表示第 i 种物品的体积、价值和数量。

输出格式

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

数据范围

0<N≤1000
0<V≤2000
0<vi,wi,si≤2000

提示:

本题考查多重背包的二进制优化方法。

输入样例

4 5
1 2 3
2 4 1
3 4 3
4 5 2

输出样例:

10
import java.io.*;
import java.util.*;class Main{static int N = 20010;static int n,V;static int[] v = new int[N]; // 体积static int[] w = new int[N]; // 价值static int[] s = new int[N]; // 个数static int[] f = new int[N]; // 二维会超内存public static void main(String[] args) throws IOException{BufferedReader in = new BufferedReader(new InputStreamReader(System.in));String[] str = in.readLine().split(" ");n = Integer.parseInt(str[0]);V = Integer.parseInt(str[1]);int cnt = 1;for(int i=1;i<=n;i++){str = in.readLine().split(" ");int vi = Integer.parseInt(str[0]);int wi = Integer.parseInt(str[1]);int si = Integer.parseInt(str[2]);// 直接将该物品数目拆分成多个,但是拆分完后的物品数目可以组合成si中的任何一个数目int k = 1;while(si>=k){ //满足c<2^(k+1) ,则是s大于k,才能划分k个物品出去v[cnt] = vi*k;w[cnt] = wi*k;si -= k;k *= 2;cnt++;}if(si!=0){ // 最后剩下的物品v[cnt] = si*vi;w[cnt] = si*wi;cnt++;}}n = cnt;// 转化为01背包问题for(int i=1;i<n;i++)for(int j=V;j>=v[i];j--)f[j] = Math.max(f[j],f[j-v[i]]+w[i]);                      System.out.println(f[V]); }
}

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

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

相关文章

Softmax 回归 + 损失函数 + 图片分类数据集【动手学深度学习v2】李沐动手学深度学习课程笔记

目录 Softmax回归 损失函数 图片分类数据集 Softmax回归从零开始实现 Softmax回归简洁实现 Softmax回归 回归和分类的区别 回归问题举例上节课的预测房价问题&#xff0c;分类问题就是对样本进行分类 回归和分类的具体区别 假设真实的类别为第i个类别&#xff08;值为1&#x…

Restful风格接口简介

参考资料&#xff1a; 个人感觉最实用的Restful风格简介 Restful简介&#xff1a; Restful风格具有如下特点&#xff1a; URL要面向资源&#xff0c;使用和资源相关的名词要在URL中体现版本号根据Http不同的方法&#xff0c;进行不同类型的资源操作&#xff08;如用Get表示查…

js【详解】event loop(事件循环/事件轮询)

event loop 是异步回调的实现原理 js 代码的执行过程 从前到后&#xff0c;一行一行执行如果某一行执行报错&#xff0c;则停止下面代码的执行先把同步代码执行完&#xff0c;再执行异步 event loop 图解 以下方代码为例&#xff1a; 第1步 将第 1 行代码放入调用栈 将要执行第…

python数据分析numpy基础之布尔数组方法sum和any和all

1 python数据分析numpy基础之布尔数组方法sum和any和all 在前面文章介绍的sum、mean、std等统计方法中&#xff0c;布尔值True被转为1&#xff0c;False被转为0。而any和all&#xff0c;则将非0转为True&#xff0c;0转为False。 1.1 sum() python的numpy库的sum()函数&…

vue element plus Icon 图标

Element Plus 提供了一套常用的图标集合。 使用图标# 如果你想像用例一样直接使用&#xff0c;你需要全局注册组件&#xff0c;才能够直接在项目里使用。 如若需要查看所有可用的 SVG 图标请查阅 element-plus/icons-vue1.xelement-plus/icons-vuelatest 和有关 Icon Collect…

C# 生成有序Guid

C# 生成有序Guid public enum SequentialGuidType {/// <summary>/// 用于 MySql 和 PostgreSql./// 当使用 <see cref"Guid.ToString()" /> 方法进行格式化时连续./// </summary>AsString,/// <summary>/// 用于 Oracle./// 当使用 <s…

JavaScript极速入门(2)

JQuery W3C标准给我们提供了一系列函数,让我们可以操作: 网页内容 网页结构 网页样式 但是原生的JavaScript提供的API操作DOM元素时,代码比较繁琐,冗长.我们学习使用JQuery来操作页面对象. JQuery是一个快速,简洁且功能丰富的JavaScript框架,于2006年发布.它封装JavaScript常…

Linux运维:实现光盘开机自动挂载、配置本地yum源教程

Linux运维&#xff1a;实现光盘开机自动挂载、配置本地yum源教程 一、光盘开机自动挂载1、检查光驱设备2、创建挂载点3、编辑/etc/fstab文件4、测试挂载 二、配置本地yum源(挂载光盘或ISO文件)1、挂载ISO文件2、创建YUM仓库配置文件3、清理YUM缓存并测试 &#x1f496;The Begi…

【netty系列-02】深入理解socket本质和BIO底层实现

Netty系列整体栏目 内容链接地址【一】深入理解网络通信基本原理和tcp/ip协议https://zhenghuisheng.blog.csdn.net/article/details/136359640【二】深入理解Socket本质和BIOhttps://zhenghuisheng.blog.csdn.net/article/details/136549478 深入理解socket本质和bio底层实现 …

C语言进阶——位段

在C语言中&#xff0c;位段&#xff08;Bit Fields&#xff09;是一种用来对结构体中的成员进行位级别的控制的特性。通过位段&#xff0c;我们可以灵活地控制结构体中各个成员的位数&#xff0c;从而节省内存空间并提高程序的效率。本篇博客将详细讲解C语言中位段的相关知识&a…

力扣爆刷第89天之hot100五连刷31-35

力扣爆刷第89天之hot100五连刷31-35 文章目录 力扣爆刷第89天之hot100五连刷31-35一、25. K 个一组翻转链表二、138. 随机链表的复制三、148. 排序链表四、23. 合并 K 个升序链表五、146. LRU 缓存 一、25. K 个一组翻转链表 题目链接&#xff1a;https://leetcode.cn/problem…

python第九节:类的使用(3)

类的继承&#xff1a; 一个类继承另一个类时&#xff0c;它将自动获得另一个类的所有属性和方法&#xff1b;原有的类称为父类&#xff0c;而新类称为子类。子类继承了其父类的所有属性和方法&#xff0c;同时还可以定义自己的属性和方法。 创建子类时&#xff0c;必须在括号…

Java 简历优化及注意事项

Java 简历优化及注意事项 前言1、自我介绍2、掌握技术技能3、项目经验4、项目必问的细节点:5、项目中的难点以及优化改进点6、获奖经历7、面试注意事项 前言 最新的 Java 面试题&#xff0c;技术栈涉及 Java 基础、集合、多线程、Mysql、分布式、Spring全家桶、MyBatis、Dubbo…

找出单身狗1,2

目录 1. 单身狗12. 单身狗2 1. 单身狗1 题目如下&#xff1a; 思路&#xff1a;一部分人可能会使用对数组排序&#xff0c;遍历数组的方式去找出只出现一次的数字&#xff0c;但这种方法的时间复杂度过高&#xff0c;有时候可能会不满足要求。 有一种十分简便的方法是使用异或…

​LeetCode解法汇总2834. 找出美丽数组的最小和

目录链接&#xff1a; 力扣编程题-解法汇总_分享记录-CSDN博客 GitHub同步刷题项目&#xff1a; https://github.com/September26/java-algorithms 原题链接&#xff1a;. - 力扣&#xff08;LeetCode&#xff09; 描述&#xff1a; 给你两个正整数&#xff1a;n 和 target …

DEAP:利用生理信号进行情绪分析的数据库【DEAP数据集】

文章目录 摘要引言刺激选择实验环境参与者步骤参与者自我评估 主观评价分析EEG频率与参与者评分之间的相关性单次试验分类结果 结论 点击下载原文 摘要 ● DEAP&#xff1a;用于分析人类情感状态的多模态数据集。 ● 32名参与者观看了40个一分钟长的音乐视频。 ● 参与者根据唤…

c++ primer中文版第五版作业第十三章

仓库地址 文章目录 13.113.213.313.413.513.613.713.813.913.1013.1113.1213.1313.1413.1513.1613.1713.1813.1913.2013.2113.2213.2313.2413.2513.2613.2713.2813.2913.3013.3113.3213.3313.3413.3513.3613.3713.3813.3913.4013.4113.4213.4313.4413.4513.4613.4713.4813.4913…

PostgreSQL教程(二十二):服务器管理(四)之服务器配置

一、设置参数 1.1 参数名称和值 所有参数名都是大小写不敏感的。每个参数都可以接受五种类型之一的值&#xff1a; 布尔、字符串、整数、 浮点数或枚举。该类型决定了设置该参数的语法&#xff1a; 布尔: 值可以被写成 on, off, true, false, yes, no, 1, 0 &#xff08;都是…

Programming Abstractions in C阅读笔记:p312-p326

《Programming Abstractions in C》学习第77天&#xff0c;p312-p326&#xff0c;总计15页&#xff0c;第7章完结。 一、技术总结 第7章主要讲算法分析——引入时间复杂度这一概念来评估算法的快慢。时间复杂度使用大O符号来表示。 第7章以排序算法为示例&#xff0c;包含&a…

go调用 c++中数组指针相关

要在Go语言中调用C编译的DLL&#xff08;动态链接库&#xff09;并传递数组&#xff0c;你需要遵循以下步骤&#xff1a; 编写C代码&#xff1a;首先&#xff0c;你需要有一个C的DLL&#xff0c;它提供了你想要在Go中调用的函数。为了确保Go可以调用它&#xff0c;你需要使用C…