动物园

题目描述

原题来自:APIO 2007

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

image

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

image

假如你将围栏 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 个小朋友会高兴。这种方法使得了最多的小朋友高兴。

输入格式

输入的第一行包含两个整数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是相同的。

输出格式

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

样例

样例输入 1
14 5
2 1 2 4 2 6
3 1 1 6 4
6 1 2 9 6 8
8 1 1 9 12
12 3 0 12 13 2
样例输出 1
5
样例说明 1

样例1给出了前面描述的示例情形。它使得所有小朋友(C=5)高兴。

样例输入 2
12 7
1 1 1 1 5
5 1 1 5 7
5 0 3 5 7 9
7 1 1 7 9
9 1 1 9 11
9 3 0 9 11 1
11 1 1 11 1
样例输出 2
6
样例说明 2

样例2给出了这样的情形,要使得所有小朋友(C=7)都高兴是不可能的。

数据范围与提示

对于每一个测试点,如果你的答案正确,则该测试点得满分,否则得0分。

对于全部数据,10<=N<=10e4,1<=C<=5*10e4,1<=E<=N

----------------------------------------------------------------------------------------------------------------------------分割线

将小朋友固定于i位置,其可见范围为[i~i+4],长度为5,可用状压思想枚举其状态

f[1~n][s]由1~n顺次枚举不同s状态,第i状态受到第i-1状态的影响

此处划重点:f[n][]中状态需要与f[1][]中i-1推出来的状态保持一致,故枚举f[1][s]的s状态时只将f[0][]中的s状态初始化0,其他均为负无穷(0x80过于大,此处选取0xcf较为合适);

f[0]数组与f[n]状态保持一直,可将f[0]看成f[n],由其推f[1]状态

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 const int maxn=1e4+10;
 4 int n,c,e,ff,l,x,Max,f[maxn][32],gx[maxn][32];//g[e][s]记为e位置的小朋友,s状态下高兴的个数
 5 void solve() {
 6     for(int s=0;s<32;s++) {
 7         memset(f,0xcf,sizeof(f));
 8         f[0][s]=0;
 9         for(int i=1;i<=n;i++) {
10             for(int j=0;j<32;j++) {
11                 f[i][j]=max(f[i-1][(j&15)<<1],f[i-1][(j&15)<<1|1])+gx[i][j];//递推式 晦涩难懂
12             }
13         }
14         Max=max(Max,f[n][s]);//f[n]起始s状态(也为f[0])高兴小朋友个数,选取高兴小朋友最多的个数
15     }
16     printf("%d
",Max);
17 }
18 int main() {
19     scanf("%d%d",&n,&c);
20     for(int i=1;i<=c;i++) {
21         scanf("%d%d%d",&e,&ff,&l);
22         int fear=0,like=0;
23         for(int j=1;j<=ff;j++) {
24             int num=0;
25             scanf("%d",&x);
26             num=(x-e+n)%n;
27             fear|=(1<<num);//1代表害怕位置
28         }
29         for(int j=1;j<=l;j++) {
30             int num=0;
31             scanf("%d",&x);
32             num=(x-e+n)%n;
33             like|=(1<<num);//1代表喜欢位置
34         }
35         for(int s=0;s<32;s++) {
36             if((s&fear)||(~s&like)) gx[e][s]++;//该s状态((1<<5)-1)种下,如果满足e位置的小朋友可观看到的五个位置:
          1.至少有一个喜欢的小动物存在 2.至少有一个害怕的小动物被移走;二者满足其一则该小朋友高兴
         判断式:(s&fear)||(~s&like)//
37 } 38 } 39 solve(); 40 return 0; 41 }

有红色的判断式就不用绿色的注释了吧=.= 今天换成紫色的注释

原文地址:https://www.cnblogs.com/1999-09-21Karry-erfs/p/13192110.html