题解 P1776 【宝物筛选_NOI导刊2010提高(02)】
为了学习单调队列优化DP奔向了此题。。。
基础的多重背包就不展开了。设
显然,
这样是
众所周知,DP优化的根本原则是去掉无用的状态、利用重复转移的状态。可是这方程一眼根本看不出什么可优化的地方啊。。。。。。
我们要像这位Dalao一样善于发现,他的blog
所以,不管这个想法是怎么来的,我们先把
然后方程就变成了这样
突然看到了
设
首先枚举
这样就是
结合代码理解会更轻松哦
#include<cstdio>
#define RG register
#define R RG int
#define G c=getchar()
const int N=1e5+9;
int f[N],g[N],q[N];
inline int in(){
RG char G;
while(c<'-')G;
R x=c&15;G;
while(c>'-')x=x*10+(c&15),G;
return x;
}
inline int max(R x,R y){return x>y?x:y;}
inline void chkmx(R&x,R y){if(x<y)x=y;}
int main(){
R n=in(),maxw=in(),maxk,lim,v,w,m,d,i,k,k1,h,t,now;
for(i=1;i<=n;++i){
v=in();w=in();m=in();
for(d=0;d<w;++d){//枚举余数
maxk=(maxw-d)/w;lim=max(maxk-m,0);//先确定最初的范围
for(t=0,k=maxk-1;k>=lim;--k){//窗口先扩大宽度到m
now=f[k*w+d]-k*v;
while(t&&g[t]<=now)--t;//维护单调性
g[++t]=now;q[t]=k;
}
for(h=1,k1=maxk;~k1;--k1,--k){//可以开始转移了
if(h<=t&&q[h]>=k1)++h;//接着移动
if(h<=t)chkmx(f[k1*w+d],g[h]+k1*v);//转移
if(k<0)continue;//注意窗口可能已经出正数范围了
now=f[k*w+d]-k*v;
while(h<=t&&g[t]<=now)--t;//维护单调性
g[++t]=now;q[t]=k;
}
}
}
printf("%d\n",f[maxw]);
return 0;
}