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

题意:

$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[N],b[N];
inline int w(int i,int j){
    return i<=0 ? 0 : (a[i]|a[j]) - (a[i]&a[j]);
}
int f[N][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);
    }
}

 

posted @ 2017-03-07 14:32  Candy?  阅读(668)  评论(0编辑  收藏  举报