4069: [Apio2015]巴厘岛的雕塑

7 篇文章 0 订阅

4069: [Apio2015]巴厘岛的雕塑

Time Limit: 10 Sec   Memory Limit: 64 MB
Submit: 344   Solved: 163
[ Submit][ Status][ Discuss]

Description

印尼巴厘岛的公路上有许多的雕塑,我们来关注它的一条主干道。
在这条主干道上一共有 N 座雕塑,为方便起见,我们把这些雕塑从 1 到 N 连续地进行标号,其中第 i 座雕塑的年龄是 Yi 年。为了使这条路的环境更加优美,政府想把这些雕塑分成若干组,并通过在组与组之间种上一些树,来吸引更多的游客来巴厘岛。
下面是将雕塑分组的规则:
这些雕塑必须被分为恰好 X 组,其中 A< = X< = B,每组必须含有至少一个雕塑,每个雕塑也必须属于且只属于一个组。同一组中的所有雕塑必须位于这条路的连续一段上。
当雕塑被分好组后,对于每个组,我们首先计算出该组所有雕塑的年龄和。
计算所有年龄和按位取或的结果。我们这个值把称为这一分组的最终优美度。
请问政府能得到的最小的最终优美度是多少?
备注:将两个非负数 P 和 Q 按位取或是这样进行计算的:
首先把 P 和 Q 转换成二进制。
设 nP 是 P 的二进制位数,nQ 是 Q 的二进制位数,M 为 nP 和 nQ 中的最大值。P 的二进制表示为 pM−1pM−2…p1p0,Q 的二进制表示为 qM−1qM−2…q1q0,其中 pi 和 qi 分别是 P 和 Q 二进制表示下的第 i 位,第 M−1 位是数的最高位,第 0 位是数的最低位。
P 与 Q 按位取或后的结果是: (pM−1  OR  qM−1)(pM−2 OR qM−2)…(p1 OR q1)(p0 OR q0)。其中:
0 OR 0=0
0 OR 1=1
1 OR 0=1
1 OR 1=1

Input

输入的第一行包含三个用空格分开的整数 N,A,B。

第二行包含 N 个用空格分开的整数 Y1,Y2,…,YN。

Output

输出一行一个数,表示最小的最终优美度。

Sample Input

6 1 3
8 1 2 1 5 4

Sample Output

11

explanation

将这些雕塑分为 2 组,(8,1,2) 和 (1,5,4),它们的和是 (11) 和 (10),最终优美度是 (11 OR 10)=11。(不难验证,这也是最终优美度的最小值。)

HINT

 子任务 1 (9 分)


1< = N< = 20

1< = A< = B< = N

0< = Yi< = 1000000000

子任务 2 (16 分)

1< = N< = 50

1< = A< = B< = min{20,N}

0< = Yi< = 10

子任务 3 (21 分)

1< = N< = 100

A=1

1< = B< = N

0< = Yi< = 20

子任务 4 (25 分)

1< = N< = 100

1< = A< = B< = N

0< = Yi< = 1000000000

子任务 5 (29 分)

1< = N< = 2000

A=1

1< = B< = N

0< = Yi< = 1000000000

Source

[ Submit][ Status][ Discuss]



反正我是不会的。。。基本照搬题解啦= =

从高位到低位枚举,看当前位能否构造出一种最高位或运算为0的方案

首先对于前四类数据,定义g[i][j]:前i个数分成j段是否能构造出符合当前位要求的方案

转移的话,若有g[k][j-1] == 1 && sum[i] - sum[k]符合本位以及先前的要求,那么g[i][j] = 1

判断的话直接与运算瞎搞一下就行了,,(很好想的)

此处复杂度O(n^3logn)


这样是过不掉第五类的

不过,,第五类A == 1,定义f[i]:构造出符合当前位的方案,最少要分多少段

转移类似。因为下界固定为1,所以段数当然是越少越容易符合题意了

此处复杂度O(n^2logn)

#include<iostream>
#include<cstring>
#include<vector>
#include<queue>
#include<algorithm>
#include<cmath>
#include<cstdio>
#include<bitset>
using namespace std;

const int maxn = 101;
const int maxm = 2002;
const int INF = ~0U>>1;
typedef long long LL;

int n,A,B,f[maxm];
bool g[maxn][maxn];
LL Ans,t,sum[maxm];

int main()
{
	#ifdef DMC
		freopen("DMC.txt","r",stdin);
	#endif
	
	cin >> n >> A >> B;
	for (int i = 1; i <= n; i++) 
		scanf("%lld",&sum[i]),sum[i] += sum[i-1];
	if (A == 1)
	{
		for (LL I = 40; I >= 0; I--)
		{
			for (int i = 1; i <= n; i++) f[i] = INF; t |= 1LL<<I;
			for (int i = 1; i <= n; i++)
				for (int j = 0; j < i; j++)
					if (f[j] != INF && !(sum[i]-sum[j]&t))
						f[i] = min(f[i],f[j] + 1);
			if (f[n] > B) Ans |= 1LL<<I,t ^= 1LL<<I;
		}
	}
	else
	{
		for (LL I = 40; I >= 0; I--)
		{
			memset(g,0,sizeof(g)); g[0][0] = 1; t |= 1LL<<I;
			for (int i = 1; i <= n; i++)
				for (int j = 1; j <= min(i,B); j++)
					for (int l = 0; l < i; l++)
						if (g[l][j-1] && !(sum[i]-sum[l]&t))
						{
							g[i][j] = 1; break;
						}
			bool pass = 0;
			for (int i = A; i <= B; i++)
				if (g[n][i]) {pass = 1; break;}
			if (!pass) Ans |= 1LL<<I,t ^= 1LL<<I;
		}
	}
	cout << Ans;
	return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值