hdu–2369 Bone Collector II(01背包变形题)
题意:求解01背包价值的第K优解。
分析:
基本思想是将每个状态都表示成有序队列,将状态转移方程中的max/min转化成有序队列的合并。
首先看01背包求最优解的状态转移方程:
\[dp\left[ j \right] = \max \left\{ {dp\left[ j \right],dp\left[ {j - a\left[ i \right].w} \right] + a\left[ i \right].v} \right\}\]如果要求第K优解,那么状态 dp[j] 就应该是一个大小为K的数组dp[j][1..K]。(其中dp[j][k]表示背包大小为j时,第k优解的值。 )
Note:
dp[j]是一个大小为K的数组,或者也可以简单地理解为在原来的方程中加了一维。
我们将利用手段维护 dp[j][1..K] 的有序性,然后原方程就可以解释为:
\[dp\left[ j \right]\left[ {1 \ldots K} \right] = merge\left\{ {dp\left[ j \right]\left[ k \right],dp\left[ {j - a\left[ i \right].w} \right]\left[ k \right] + a\left[ i \right].v|0 \le k \le K} \right\}\]有序队列 dp[j-a[i].w]+a[i].v 则理解为在 dp[j-a[i].w][1..K] 的每个数上加上 a[i].v 后得到的有序队列。
合并这两个有序队列并将结果的前K项储存到dp[j][1..K]中的复杂度是O(K), 最后的答案是f[N] [V][K]。总的复杂度是O(VNK)。
---以上思路摘自 第K优解问题
【问题】如何合并两个队列?
已知队列P[1…K],Q[1…K],均为从大到小的顺序排列,求解队列R[1…K]是P,Q合并后的前K项。
首先,我们得考虑去重问题,故我们不能将P,Q队列合并再排序,取前K项了事。
我曾借助于STL中的SET,利用SET中的元素互异性解决去重问题,但很可惜超时。
精巧的解决方案:
//队列名为:alpha[1...K],beta[1...K]
alpha[k+1]=-1;
beta[k+1]=-1;
int t=1,p=1,q=1;
while(t<=k &&(p<=k && q<=k)){
if(alpha[p]>beta[q]) dp[j][t]=alpha[p++];
else dp[j][t]=beta[q++];
if(dp[j][t]!=dp[j][t-1]) t++;
}Note:
- 这里要注意(p<=k || q<=k)不能写成(p<=k && q<=k),因为某一个队列取完元素,并不等价于我们取到了前K个元素。
- 我们需要初始化操作 alpha[K+1]=beta[K+1]=-1,只要初始化为负数均可。
我们知道背包中的解最小为0,不可能取负数,现在我们假设队列beta的下标q=k,且beta[p]=0,
表示队列beta已经取完了有效元素,而队列alpha中的下标p<k,但是alpha[p]=0.
那么下一轮循环中,条件if(alpha[p]>beta[q])不成立,会导致q++,数组下标溢出,导致循环提前异常退出,显然会导致结果出错。
所以我们这里必须初始化为负数,不可省略这一步。
- 同时为了达到去重效果,我们需要将t下标从1开始计数,否则我们将找不到dp[j][t-1],导致程序异常终止,只有当前后两个数各异时,我们才移动下标t。
解答代码:
#include<string.h>
#include<cstdio>
#include<algorithm>
using namespace std;
#define maxn_n 105
#define maxn_v 1005
#define maxn_k 35
int dp[maxn_v][maxn_k];
struct bone{
int v,w;
};
bone b[maxn_n];
int alpha[maxn_k],beta[maxn_k]; int main(){
//freopen("in.txt","r",stdin);
int cases;
scanf("%d",&cases);
while(cases--){
int n,v,k;
scanf("%d %d %d",&n,&v,&k);
for(int i=1;i<=n;i++)
scanf("%d",&b[i].v);
memset(dp,0,sizeof(dp));
for(int i=1;i<=n;i++){
scanf("%d",&b[i].w);
for(int j=v;j>=b[i].w;j--){
for(int t=1;t<=k;t++){
alpha[t]=dp[j][t];
beta[t]=dp[j-b[i].w][t]+b[i].v;
}
alpha[k+1]=-1;
beta[k+1]=-1;
int t=1,p=1,q=1;
while(t<=k &&(p<=k ||q<=k)){
if(alpha[p]>beta[q]) dp[j][t]=alpha[p++];
else dp[j][t]=beta[q++];
if(dp[j][t]!=dp[j][t-1]) t++;
}
}
}
printf("%d\n",dp[v][k]);
}
}
hdu–2369 Bone Collector II(01背包变形题)的更多相关文章
- HDU 2639 Bone Collector II(01背包变形【第K大最优解】)
Bone Collector II Time Limit: 5000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others ...
- HDU 2639 Bone Collector II (01背包,第k解)
题意: 数据是常规的01背包,但是求的不是最大容量限制下的最佳解,而是第k佳解. 思路: 有两种解法: 1)网上普遍用的O(V*K*N). 2)先用常规01背包的方法求出背包容量限制下能装的最大价值m ...
- HDU 2639 Bone Collector II(01背包变型)
此题就是在01背包问题的基础上求所能获得的第K大的价值. 详细做法是加一维去推当前背包容量第0到K个价值,而这些价值则是由dp[j-w[ i ] ][0到k]和dp[ j ][0到k]得到的,事实上就 ...
- HDU - 2639 Bone Collector II (01背包第k大解)
分析 \(dp[i][j][k]\)为枚举到前i个物品,容量为j的第k大解.则每一次状态转移都要对所有解进行排序选取前第k大的解.用两个数组\(vz1[],vz2[]\)分别记录所有的选择情况,并选择 ...
- HDOJ(HDU).2602 Bone Collector (DP 01背包)
HDOJ(HDU).2602 Bone Collector (DP 01背包) 题意分析 01背包的裸题 #include <iostream> #include <cstdio&g ...
- hdu 2602 Bone Collector(01背包)
题意:给出包裹的大小v,然后给出n块骨头的价值value和体积volume,求出一路下来包裹可以携带骨头最大价值 思路:01背包 1.二维数组(不常用 #include<iostream> ...
- hdu 2602 Bone Collector(01背包)模板
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2602 Bone Collector Time Limit: 2000/1000 MS (Java/Ot ...
- 题解报告:hdu 2602 Bone Collector(01背包)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2602 Problem Description Many years ago , in Teddy’s ...
- hdu 2602 - Bone Collector(01背包)解题报告
Bone Collector Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) T ...
随机推荐
- Web Audio API 实现音频可视化
声明:本文为原创文章,如需转载,请注明来源WAxes,谢谢! 一转眼就已经有三个月没写博客了,毕业季事情确实多,现在也终于完全毕业了,博客还是不能落下.偶尔还是要写一下. 玩HTML5的Audio A ...
- Bootstrap系列 -- 13. 内联表单
有时候我们需要将表单的控件都在一行内显示.在Bootstrap框架中实现这样的表单效果是轻而易举的,你只需要在<form>元素中添加类名“form-inline”即可 如果你要在input ...
- MATLAB函数freqz()
MATLAB提供了专门用于求离散系统频响特性的函数freqz(),调用freqz()的格式有以下两种:l [H,w]=freqz(B,A,N) B和A分别为离散系统的系统函数分子.分母多项式的系数向量 ...
- app状态监听广播
手机中的应用在安装.更新和卸载时都会发送广播 清单文件 <?xml version="1.0" encoding="utf-8"?> <man ...
- UEFI与MBR区别
EFI与MBR启动的区别 大硬盘和WIN8系统,让我们从传统的BIOS+MBR模式升级到UEFI+GPT模式,现在购买的主流电脑,都是预装WIN8系统,为了更好的支持2TB硬盘 ,更快速的启动win ...
- HTML5基础知识(3)--required属性
1.required属性规定在提交之前要填写输入域(不能为空). 2.代码 <body> <form> 账号:<input type="text" r ...
- sqlite之聚合函数的使用
聚合函数对一组值执行计算并返回单一的值.聚合函数对一组值执行计算,并返回单个值.除了 COUNT 以外,聚合函数都会忽略空值. 聚合函数经常与 SELECT 语句的 GROUP BY 子句一起使用. ...
- 【CodeVS 2822】爱在心中
“每个人都拥有一个梦,即使彼此不相同,能够与你分享,无论失败成功都会感动.爱因为在心中,平凡而不平庸,世界就像迷宫,却又让我们此刻相逢Our Home.” 在爱的国度里有N个人,在他们的心中都有着一个 ...
- 【线性规划与网络流 24题】已完成(3道题因为某些奇怪的原因被抛弃了QAQ)
写在前面:SDOI2016 Round1滚粗后蒟蒻开始做网络流来自我拯救(2016-04-11再过几天就要考先修课,现在做网络流24题貌似没什么用←退役节奏) 做的题目将附上日期,见证我龟速刷题. 1 ...
- Integer与int的种种比较你知道多少?
如果面试官问Integer与int的区别:估计大多数人只会说道两点,(1)Ingeter是int的包装类,int的初值为0:(2)Ingeter的 初值为null.但是如果面试官再问一下Integer ...