题目描述
题目链接
https://www.acwing.com/problem/content/5/
思路:
多重背包的模板题目,考察二进制优化,单调队列优化。本质上就是把dp转移式子写出来,然后维护一个单调递减的队列去O(1)查找最大值即可。
代码
#include<iostream>
#include<cstdio>
using namespace std;
const int N=2010;
int f[N][N];
int n,m;
int s[N],v[N],w[N];
int q[N];
int tt,hh;
int main(void){
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++) scanf("%d%d%d",&v[i],&w[i],&s[i]);
for(int i=1;i<=n;i++){
for(int r=0;r<v[i];r++){
tt=hh=0;
for(int j=0;v[i]*j+r<=m;j++){
int cur=r+v[i]*j;
int cur_value=f[i-1][cur]-j*w[i];
while(tt!=hh){
int g=f[i-1][q[tt-1]]-(q[tt-1]-r)/v[i]*w[i];
if(g<=cur_value) tt--;
else break;
}
q[tt++]=cur;
if(cur-q[hh]>s[i]*v[i]) hh++;
f[i][cur]=f[i-1][q[hh]]+(cur-q[hh])/v[i]*w[i];
}
}
}
printf("%d",f[n][m]);
return 0;
}