说说多重背包的循环次序

2023-12-16 07:08
文章标签 背包 循环 多重 次序

本文主要是介绍说说多重背包的循环次序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

鉴于多次被体积和次数的循环里外顺序坑,就决定今天跟你好好掰吃掰吃!!!!
1.首先,想说的是,普通的多重背包写法,是相当于每取1次物品就更新一遍体积的值,这就像是01背包一样
要保证当次循环是没更新过的值,所以体积不仅要倒过来,还得要体积在外面,因为
HUA重点!!!!当前层的物品只能用上一层的物品去更新!!!就是这件物品,只能由上一个物品转移过来,
而不是同种物品的不同取法,因为同种物品,只能有一种取法,你总不能拿2次又拿3次吧!!
2.其次,二进制的写法,不多说怎么写了,简单说说为什么这么写。

   for (int k = 1; k <= s; k <<= 1){ve.push_back(k);s -= k;}

每次取2的几次幂个,有的小伙伴不明白为什么要s -= k,比如说我,我刚拿过来这个循环第一想法是
我草!!这尼玛,为什么这s这么牛逼,还能减!!现在回过头来再写这个的时候,只能想当时还是挺脑残的
因为你既然拿了这么多次了,当然就要给人家减下去那么多啊,总不能拿了人家100块钱,人家账户余额
还原封不动吧!还要注意的是跟完全背包不一样的是,体积仍然是倒着的,因为这个看成很多个01背包!不能无限拿!
这个思想就是拆分,因为每种取法,都能有以上几种取法凑出来,所以说当前层的物品需要同种物品的值,
这就相当于你能拿很多次同种物品,仔细体会以上两种写法的区别哦!
3.单调队列,这种写法没什么好说的,因为枚举形式和普通的一样,不是拼凑,所以说不说了!
就酱紫,说实话,俩月之后回来在复习背包问题,发现不像原来想的那么复杂,更不像原来想的那么玄学
每层循环都是有原因的,这就相当于模拟你的决策过程,你想怎么写就怎么写,但要遵守游戏规则。
希望下次回来,我还能犯这些错误!!!!!哈哈哈,加油!!!再次勉励自己一下!!!
也希望不管闲着无聊还是电脑卡顿错点进来看到我写的这些废话,能给更多人带来启发,我想我没有白白反思!
希望全世界的编程爱好者都go go go!
有一道混合背包的题:
https://www.acwing.com/problem/content/7/

#include <iostream>
#include <algorithm>
using namespace std;
const int maxn = 1e3 + 7;
int dp[maxn], v, w, s, m;
int main(){int n, m;cin >> n >> m;for (int i = 1; i <= n; i++){cin >> v >> w >> s;if(s == -1){for (int j = m; j >= v; j--)dp[j] = max(dp[j], dp[j - v] + w);}else if(s == 0){for (int j = v; j <= m; j++)dp[j] = max(dp[j], dp[j - v] + w);}else{vector<int>ve;for (int k = 1; k <= s; k <<= 1){ve.push_back(k);s -= k;}if(s)ve.push_back(s);for (auto t : ve){for (int j = m; j >= 0; j--)//这两个循环不能倒过来!if(t*v <= j)dp[j] = max(dp[j], dp[j - t*v] + t*w);}}}printf("%d\n",dp[m]);return 0;
}

这篇关于说说多重背包的循环次序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/499578

相关文章

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

poj2576(二维背包)

题意:n个人分成两组,两组人数只差小于1 , 并且体重只差最小 对于人数要求恰好装满,对于体重要求尽量多,一开始没做出来,看了下解题,按照自己的感觉写,然后a了 状态转移方程:dp[i][j] = max(dp[i][j],dp[i-1][j-c[k]]+c[k]);其中i表示人数,j表示背包容量,k表示输入的体重的 代码如下: #include<iostream>#include<

hdu2159(二维背包)

这是我的第一道二维背包题,没想到自己一下子就A了,但是代码写的比较乱,下面的代码是我有重新修改的 状态转移:dp[i][j] = max(dp[i][j], dp[i-1][j-c[z]]+v[z]); 其中dp[i][j]表示,打了i个怪物,消耗j的耐力值,所得到的最大经验值 代码如下: #include<iostream>#include<algorithm>#include<

csu(背包的变形题)

题目链接 这是一道背包的变形题目。好题呀 题意:给n个怪物,m个人,每个人的魔法消耗和魔法伤害不同,求打死所有怪物所需的魔法 #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>//#include<u>#include<map

hdu1011(背包树形DP)

没有完全理解这题, m个人,攻打一个map,map的入口是1,在攻打某个结点之前要先攻打其他一个结点 dp[i][j]表示m个人攻打以第i个结点为根节点的子树得到的最优解 状态转移dp[i][ j ] = max(dp[i][j], dp[i][k]+dp[t][j-k]),其中t是i结点的子节点 代码如下: #include<iostream>#include<algorithm

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

hdu 2602 and poj 3624(01背包)

01背包的模板题。 hdu2602代码: #include<stdio.h>#include<string.h>const int MaxN = 1001;int max(int a, int b){return a > b ? a : b;}int w[MaxN];int v[MaxN];int dp[MaxN];int main(){int T;int N, V;s

uva 10130 简单背包

题意: 背包和 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#include <stack>#include <vector>#include <queue>#include <map>

HDU 2159 二维完全背包

FATE 最近xhd正在玩一款叫做FATE的游戏,为了得到极品装备,xhd在不停的杀怪做任务。久而久之xhd开始对杀怪产生的厌恶感,但又不得不通过杀怪来升完这最后一级。现在的问题是,xhd升掉最后一级还需n的经验值,xhd还留有m的忍耐度,每杀一个怪xhd会得到相应的经验,并减掉相应的忍耐度。当忍耐度降到0或者0以下时,xhd就不会玩这游戏。xhd还说了他最多只杀s只怪。请问他能

多重背包转换成0-1背包

http://acm.hdu.edu.cn/showproblem.php?pid=2191 多重背包特点: 一种物品有C个(既不是固定的1个,也不是无数个) 优化的方法: 运用神奇的二进制,进行物品拆分,转化成01背包 物品拆分,把13个相同的物品分成4组(1,2,4,6) 用这4组可以组成任意一个1~13之间的数! 原理:一个数总可以用2^