4069: [Apio2015]巴厘岛的雕塑

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

题解:

对于n<=200的:

从高到低枚举所有位,要使得对于每组该位最好都为0。

令dp[i][j]表示前i棵树分为j组是否能够满足条件。 枚举k,表示令k+1~j为一组。

满足条件当且仅当dp[k][j-1]满足条件且k+1~j为一组时满足当前所有位的条件。

时间复杂度为n^3*位数。

对于A=1的:

仍然从高到低枚举每一位。由于A=1的存在,我们可以令dp[i]表示前i个数在满足条件的情况下最少分为几组。

因此复杂度降为n^2*位数。

code:

 1 #include<cstdio>
 2 #include<iostream>
 3 #include<cmath>
 4 #include<cstring>
 5 #include<algorithm>
 6 using namespace std;
 7 typedef long long int64;
 8 char ch;
 9 bool ok;
10 void read(int &x){
11     for (ok=0,ch=getchar();!isdigit(ch);ch=getchar()) if (ch==‘-‘) ok=1;
12     for (x=0;isdigit(ch);x=x*10+ch-‘0‘,ch=getchar());
13     if (ok) x=-x;
14 }
15 const int maxn=2005;
16 int n,a,b,v[maxn],g[maxn];
17 int64 sum[maxn];
18 bool f[205][205];
19 bool check(bool flag,int64 ans){
20     if (flag){
21         memset(g,63,sizeof(g));
22         g[0]=0;
23         for (int i=1;i<=n;i++) for (int j=0;j<i;j++) if (((sum[i]-sum[j])|ans)==ans) g[i]=min(g[i],g[j]+1);
24         return g[n]<=b;
25     }
26     else{
27         memset(f,0,sizeof(f));
28         f[0][0]=1;
29         for (int i=1;i<=n;i++) for (int j=1;j<=b;j++) for (int k=0;k<i;k++) if (((sum[i]-sum[k])|ans)==ans) f[i][j]|=f[k][j-1];
30         for (int i=a;i<=b;i++) if (f[n][i]) return true;
31         return false;
32     }
33 }
34 int main(){
35     read(n),read(a),read(b);
36     for (int i=1;i<=n;i++) read(v[i]),sum[i]=sum[i-1]+v[i];
37     int64 ans=(~0ULL>>1);
38     for (int i=62;i>=0;i--){
39         ans^=1LL<<i;
40         if (!check(a==1,ans)) ans^=1LL<<i;
41     }
42     printf("%lld\n",ans);
43     return 0;
44 }
时间: 2024-10-27 02:30:46

4069: [Apio2015]巴厘岛的雕塑的相关文章

bzoj 4069 [Apio2015]巴厘岛的雕塑 dp

[Apio2015]巴厘岛的雕塑 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 494  Solved: 238[Submit][Status][Discuss] Description 印尼巴厘岛的公路上有许多的雕塑,我们来关注它的一条主干道. 在这条主干道上一共有 N 座雕塑,为方便起见,我们把这些雕塑从 1 到 N 连续地进行标号,其中第 i 座雕塑的年龄是 Yi 年.为了使这条路的环境更加优美,政府想把这些雕塑分成若干组,并通过在组与组之间

【BZOJ 4069】[apio2015]巴厘岛的雕塑

4069:[apio2015]巴厘岛的雕塑 Time limit: 1000 ms Memory limit: 65536 KB Description The province of Bali has many sculptures located on its roads. Let's focus on one of its main roads. There are N sculptures on that main road, conveniently numbered 1 throug

bzoj千题计划239:bzoj4069: [Apio2015]巴厘岛的雕塑

http://www.lydsy.com/JudgeOnline/problem.php?id=4069 a!=1: 从高位到低位一位一位的算 记录下哪些位必须为0 dp[i][j] 表示前i个数分为j组,这一位为0,且满足之前必须为0的位也是0 是否可行 枚举k,表示k+1~i分为一组 若k+1~i的和满足 必须为0的位是0,且dp[k][j-1] 为true 则dp[i][j]为true a=1: 从高位到低位一位一位的算 记录下哪些位必须为0 dp[i] 表示前i个数,这一位为0,且满足之

【BZOJ4069】[Apio2015]巴厘岛的雕塑 按位贪心+DP

[BZOJ4069][Apio2015]巴厘岛的雕塑 Description 印尼巴厘岛的公路上有许多的雕塑,我们来关注它的一条主干道. 在这条主干道上一共有 N 座雕塑,为方便起见,我们把这些雕塑从 1 到 N 连续地进行标号,其中第 i 座雕塑的年龄是 Yi 年.为了使这条路的环境更加优美,政府想把这些雕塑分成若干组,并通过在组与组之间种上一些树,来吸引更多的游客来巴厘岛. 下面是将雕塑分组的规则: 这些雕塑必须被分为恰好 X 组,其中 A< = X< = B,每组必须含有至少一个雕塑,每

[APIO2015]巴厘岛的雕塑

刚开始看错题,以为是每一组或起来,求和的最小值,然后就瞎写了个dp,状态根本无法转移竟然过了六个点也是醉了 这个题根据数据可以分成两部分,可以按位贪心检验 对于n<=100的数据每一位直接n^3的dp就可以了 然后对于n>100的数据,A相当于没有限制,对每一位可以直接搞, g[i]代表到第i个雕塑至少分多少组才能符合要求 1 #define MAXN 2010UL 2 #include <cstdio> 3 #include <cstring> 4 5 using n

APIO2015巴厘岛的雕塑——数位DP

题目:https://www.luogu.org/problemnew/show/P3646 对于A>1,将答案各位全置1,然后从高位到低位改成0判断是否可行: 用f[i][j]数组代表前i个数分成j组是否可行,转移是枚举最后一段的左端点k,然后看看后面整个一段的和能否满足要求,如果前后都满足就表示i,j状态也可行: 对于A=1,可以贪心地认为分组数量越少越好,所以可行性转化为最优性,省去一维,转移条件同上,取min即可: 先写了个WA一半的版本: #include<iostream>

[APIO2015]巴厘岛的雕塑[按位贪心+dp]

题意 给你长度为 \(n\) 的序列,要求分成 \(k\) 段连续非空的区间,求所有区间和的 \(or\) 最小值. 分析 定义 \(f_{i,j}\) 表示前 \(i\) 个点分成 \(j\) 段的最小 \(or\) 是有问题的,因为可能有一位一定在后面出现而某一位没有必要在后面出现,这时贪心就出现了问题. 考虑按位贪心,假设考虑到了第 \(b\) 位,定义 \(f_{i,j}\) 表示前 \(i\) 个点分成 \(j\) 段且满足 \(b\) 以上位的限制,第 \(b\) 位为 \(0\)

bzoj4069: [Apio2015]巴厘岛的雕塑

暴力三维DP容易想到 也容易想到贪心每一位,尽量不选 但是怎么check呢?又是拼起来 f[i][j]表示枚举到第i位,分成j组,且可以满足当前假设是否有可行解 转移不难自己看代码吧 但是这样是O(logm*n^3)的,subtest5过不去 但是它又有一个A==1,就可以维护枚举到第i位分成组数最小的方案数 好题啊! #include<cstdio> #include<iostream> #include<cstring> #include<cstdlib>

bzoj4069【APIO2015】巴厘岛的雕塑

4069: [Apio2015]巴厘岛的雕塑 Time Limit: 10 Sec  Memory Limit: 64 MB Submit: 192  Solved: 89 [Submit][Status][Discuss] Description 印尼巴厘岛的公路上有许多的雕塑,我们来关注它的一条主干道. 在这条主干道上一共有 N 座雕塑,为方便起见,我们把这些雕塑从 1 到 N 连续地进行标号,其中第 i 座雕塑的年龄是 Yi 年.为了使这条路的环境更加优美,政府想把这些雕塑分成若干组,并通