bzoj1151: [CTSC2007]动物园zoo 状压dp

Description

新建的圆形动物园是亚太地区的骄傲。圆形动物园坐落于太平洋的一个小岛上,包含一大圈围栏,每个围栏里有一
种动物。如下图所示:

你是动物园的公共主管。你要做的是,让每个来动物园的人都尽可能高兴。今天有一群小朋友来动物园参观,你希
望能让他们在动物园度过一段美好的时光。但这并不是一件容易的事——有的动物有一些小朋友喜欢,有的动物有
一些小朋友害怕。如,Alex 喜欢可爱的猴子和考拉,而害怕拥牙齿锋利的狮子。而Polly 会因狮子有美丽的鬃毛
而喜欢它,但害怕有臭味的考拉。你可以选择将一些动物从围栏中移走以使得小朋友不会害怕。但你不能移走所有
的动物,否则小朋友们就没有动物可看了。每个小朋友站在大围栏圈的外面,可以看到连续的 5 个围栏。你得到
了所有小朋友喜欢和害怕的动物信息。当下面两处情况之一发生时,小朋友就会高兴:
至少有一个他害怕的动物被移走
至少有一个他喜欢的动物没被移走
例如,考虑下图中的小朋友和动物:

假如你将围栏 4 和 12 的动物移走。Alex 和 Ka-Shu 将很高兴,因为至少有一个他们害怕的动物被移走了。这也
会使 Chaitanya 高兴,因为他喜欢的围栏 6 和8 中的动物都保留了。但是,Polly 和 Hwan 将不高兴,因为他们
看不到任何他们喜欢的动物,而他们害怕的动物都还在。这种安排方式使得三个小朋友高兴。现在,换一种方法,
如果你将围栏 4 和 6 中的动物移走,Alex 和 Polly 将很高兴,因为他们害怕的动物被移走了。Chaitanya 也会
高兴,虽然他喜欢的动物 6被移走了,他仍可以看到围栏 8 里面他喜欢的动物。同样的 Hwan 也会因可以看到自
己喜欢的动物 12 而高兴。唯一不高兴的只有 Ka-Shu。如果你只移走围栏 13 中的动物,Ka-Shu 将高兴,因为有
一个他害怕的动物被移走了,Alex, Polly, Chaitanya 和 Hwan 也会高兴,因为他们都可以看到至少一个他们喜
欢的动物。所以有 5 个小朋友会高兴。这种方法使得了最多的小朋友高兴。

Input

输入的第一行包含两个整数N, C,用空格分隔。
N是围栏数(10≤N≤10 000),C是小朋友的个数(1≤C≤50 000)。
围栏按照顺时针的方向编号为1,2,3,…,N。
接下来的C行,每行描述一个小朋友的信息,
以下面的形式给出: E F L X1 X2 … XF Y1 Y2 … YL
其中: E表示这个小朋友可以看到的第一个围栏的编号(1≤E≤N),
换句话说,该小朋友可以看到的围栏为E, E+1, E+2, E+3, E+4。
注意,如果编号超过N将继续从1开始算。
如:当N=14, E=13时,这个小朋友可以看到的围栏为13,14,1, 2和3。
F表示该小朋友害怕的动物数。
L表示该小朋友喜欢的动物数。
围栏X1, X2, …, XF 中包含该小朋友害怕的动物。
围栏Y1, Y2, …, YL 中包含该小朋友喜欢的动物。
X1, X2, …, XF, Y1, Y2, …, YL是两两不同的整数,
而且所表示的围栏都是该小朋友可以看到的。
小朋友已经按照他们可以看到的第一个围栏的编号从小到大的顺序排好了
(这样最小的E对应的小朋友排在第一个,最大的E对应的小朋友排在最后一个)。
注意可能有多于一个小朋友对应的E是相同的。

Output

仅输出一个数,表示最多可以让多少个小朋友高兴

题解

看数据范围,只有小朋友看到的栏杆范围可以状压。
然后我们用\(num[i]][j]\)表示观察范围为\(i-(i+4)\)的小朋友在状态为\(j\)的情况下的高兴人数。
接着用\(f[i][j]\)表示观察范围为\(i-(i+4)\),状态为\(j\)最多的小朋友高兴人数,然后由第\(i-1\)个围栏去不去掉转移

\[f[i][j]=max(f[i-1][(j\&15)<<1],f[i-1][(j\&15)<<1|1])+num[i][j] \]

又因为它是一个环,我们枚举初始的状态,只有结尾处的状态与初始状态相同的才合法,来更新答案。
然后记得每次枚举答案时给别的状态致负值,防止影响答案。

代码

#include<iostream>
#include<cstdio> 
#include<cstring>
#include<algorithm> 
using namespace std;
const int maxn=1e4+10;
const int maxm=(1<<5)+10;
const int INF=0x3f3f3f3f;
int n,c,ans;
int num[maxn][maxm];
int f[maxn][maxm];
int id[maxn];
int main(){
    ios::sync_with_stdio(false);
    cin>>n>>c;
    for(int i=1;i<=c;i++){
        int e,a,f,l,x=0,y=0,ht,lv;
        cin>>e>>f>>l;a=e;
        for(int i=1;i<=5;i++){
            if(e>n)e=1;
            id[e++]=i;
        }
        for(int i=1;i<=f;i++){
            cin>>ht;ht=id[ht];
            x|=1<<(ht-1);
        }
        for(int i=1;i<=l;i++){
            cin>>lv;lv=id[lv];
            y|=1<<(lv-1);
        }
        for(int i=0;i<(1<<5);i++)
            if((i&y)||((i^((1<<5)-1))&x))
                num[a][i]++;
    }
    for(int i=0;i<(1<<5);i++){
        memset(f[0],-INF,sizeof(f[0]));
        f[0][i]=0;
        for(int j=1;j<=n;j++) for(int k=0;k<(1<<5);k++)
            f[j][k]=max(f[j-1][(k&15)<<1],f[j-1][(k&15)<<1|1])+num[j][k];
        ans=max(f[n][i],ans);
    }
    printf("%d\n",ans);
}
posted @ 2018-10-06 22:14  南城ㄱ  阅读(136)  评论(0编辑  收藏  举报