luogu3188/bzoj1190 梦幻岛宝珠 (分层背包dp)

他都告诉你能拆了 那就拆呗。把每个重量拆成$a*2^b$的形式

然后对于每个不同的b,先分开做30个背包

再设f[i][j]表示b<=i的物品中 容量为$ j*2^i+W\&((1<<(i-1))-1) $(就是这一位是j+W的前i-1位)的最大权值(这个容量没必要填满)

然后f[i][j]就可以从f[i-1][j*2+W的第i-1位]转移过来,再拿着这个去更新本层的其他容量

最后答案就是f[x][1],x是W的最高位1的位数

 1 #include<bits/stdc++.h>
 2 #define pa pair<int,int>
 3 #define CLR(a,x) memset(a,x,sizeof(a))
 4 using namespace std;
 5 typedef long long ll;
 6 const int maxn=110,maxs=1e3+10,maxp=33;
 7 
 8 inline ll rd(){
 9     ll x=0;char c=getchar();int neg=1;
10     while(c<'0'||c>'9'){if(c=='-') neg=-1;c=getchar();}
11     while(c>='0'&&c<='9') x=x*10+c-'0',c=getchar();
12     return x*neg;
13 }
14 
15 int N,W,sum[maxs];ll f[maxp][maxs];
16 
17 int main(){
18     //freopen(".in","r",stdin);
19     int i,j,k;
20     while(1){
21         N=rd(),W=rd();
22         if(N==-1) break;
23         CLR(f,0);CLR(sum,0);
24         int x=log2(W);
25         for(i=1;i<=N;i++){
26             int a=rd(),b=0,v=rd();
27             while(a&&a%2==0) b++,a>>=1;
28             if(b>x) continue;
29             for(j=min(sum[b],1000-a);j>=0;j--){
30                 f[b][j+a]=max(f[b][j+a-1],max(f[b][j+a],f[b][j]+v));
31             }
32             sum[b]+=a;
33         }
34         
35         for(i=0;i<x;i++){
36             for(j=1;j<=1000;j++) f[i][j]=max(f[i][j-1],f[i][j]);
37             for(k=1000;k>=0;k--){
38                 if(k&&!f[i+1][k]) continue; 
39                 for(j=1000;j>=k;j--){
40                     f[i+1][j]=max(f[i+1][j],f[i+1][k]+f[i][min(1000,2*(j-k)+((W>>i)&1))]);
41                 }
42             }
43         }
44         printf("%lld\n",f[x][1]);
45     }
46     return 0;
47 }

 

posted @ 2018-10-27 22:14  Ressed  阅读(152)  评论(0编辑  收藏  举报