BZOJ 1151: [CTSC2007]动物园zoo

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

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

Sample Input

10 10
1 1 1 3 2
2 1 0 4
3 1 1 5 6
4 1 1 7 6
5 1 0 6
6 1 2 9 8 10
7 1 0 10
8 1 0 8
9 1 1 1 2
10 1 0 2

Sample Output

10

 

 

裸的状压dp

注意这是一个环,枚举前4个

注意初始化

注意初始化

注意初始化

#include<cstdio>
#include<iostream>
using namespace std;
inline int read()
{
	int ret=0;
	char ch=getchar();
	while(ch<'0'||ch>'9') ch=getchar();
	while(ch>='0'&&ch<='9')
		ret=(ret<<1)+(ret<<3)+ch-'0',ch=getchar();
	return ret;
}

int n,m,ans;
const int  N=1e4+5,S=50;
int w[N][S],f[N][S];

int main()
{
	n=read(),m=read();
	while(m--)
	{
		int loc=read(),x=read(),y=read(),a=0,b=0;
		while(x--) a|=1<<((read()+n-loc)%n);
		while(y--) b|=1<<((read()+n-loc)%n);
		for(int i=0;i<=31;i++)
			if(i&b||(~i)&a) w[loc][i]++;
	}
	ans=0;
	for(int i=0;i<=15;i++)
	{
		for(int j=0;j<=n;j++)
			for(int k=0;k<=31;k++) f[j][k]=-1e9;
		f[1][i]=w[1][i];
		f[1][i+16]=w[1][i+16];
		for(int j=2;j<=n;j++)
			for(int k=0;k<=31;k++)
				f[j][k]=max(f[j-1][(k&15)<<1],f[j-1][((k&15)<<1)|1])+w[j][k];
		ans=max(ans,max(f[n][i<<1],f[n][(i<<1)|1]));
	}
	printf("%d\n",ans);
	return 0;
}

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值