您的位置:首页 > 其它

BZOJ 1226: [SDOI2009]学校食堂Dining [DP 状压]

2017-03-07 14:32 453 查看
题意:

$n$个人排队打饭,第$i$个人口味$a_i$,能容忍最多身后第$b_i$个人先打饭。

先后两人$i,j$做饭时间为$a_i & a_j - a_i | a_j$

求最少时间

一开始想$f[i][s]$表示第$i$个人身后人吃饭集合$s$,第$i$个人最后吃完的状态,发现没法转移

这时候应该考虑给状态加维

$f[i][s][k]$表示前$i-1$人吃完,$i$身后包括$i$的吃饭集合为$s$,最后一个吃完的人是$k$的最短时间

如果$i$吃完了,可以直接转移到$i+1$;否则枚举下一个谁吃饭,注意满足容忍度要求

还有一点重要:$f[i]$的时候集合$s$是可以出现大于$b[i]$的人吃完饭的,因为$s$中的$i$有可能已经吃完饭了

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
typedef long long ll;
const int N=1005,S=(1<<8)+5,INF=1e9;
inline int read(){
char c=getchar();int x=0,f=1;
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
return x*f;
}
int n,a
,b
;
inline int w(int i,int j){
return i<=0 ? 0 : (a[i]|a[j]) - (a[i]&a[j]);
}
int f
[S][16],Z=8;
int main(){
freopen("in","r",stdin);
int T=read();
while(T--){
n=read();
for(int i=1;i<=n;i++) a[i]=read(),b[i]=min(read(),n-i);
memset(f,127,sizeof(f));
f[1][0][Z-1]=0;
for(int i=1;i<=n;i++){
int All=1<<8;
for(int s=0;s<All;s++)
for(int k=-8;k<=7;k++) if(f[i][s][k+Z]<INF){
int now=f[i][s][k+Z];
if(s&1) {int &t=f[i+1][s>>1][k-1+Z];t=min(t,now);continue;}
int lim=7;
for(int j=0;j<=lim;j++) if( ((1<<j)&s)==0 ){
int &t=f[i][s|(1<<j)][j+Z];
t=min(t,now+w(i+k,i+j));
lim=min(lim,j+b[i+j]);
}
}
}
int ans=INF;
for(int k=-8;k<=-1;k++) ans=min(ans,f[n+1][0][k+Z]);
printf("%d\n",ans);
}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: