[2017-2018 ACM-ICPC, NEERC, Southern Subregional Contest][Codeforces 883 M]Quadcopter Competition
[Codeforces Round #445][Codeforces 886C]Petya and Catacombs

[Educational Codeforces Round 32][Codeforces 888E]Maximum Subsequence

Zarxdy34 posted @ 2017年11月11日 15:23 in Codeforces with tags 二分 , 346 阅读

    题意:给出\(a[1]\cdot a[n]\),求一个子序列\({b_n}\),最大化 \(sum=\sum_{}^{}b_n\ mod\ m\)。\(1<=n<=35,1<=m<=10^9\)

    题解:将序列分成左右两半,分别求出所有可能的和,然后对其中一个求出的和序列的所有元素,在另一个和序列二分查找出与其相加结果最大的值。

 

#include <bits/stdc++.h>
using namespace std;
const int maxn=1<<18;

int a[maxn],b[maxn],an,bn;
int arr[maxn];
int n,m,ans;

int main()
{
	scanf("%d%d",&n,&m);
	for (int i=1;i<=n;i++) scanf("%d",&arr[i]);
	an=bn=1;
	for (int i=1;i<=n/2;i++)
	{
		for (int j=0;j<an;j++) a[j+an]=(a[j]+arr[i])%m;
		sort(a,a+an*2);
		an=unique(a,a+an*2)-a;
	}
	for (int i=n/2+1;i<=n;i++)
	{
		for (int j=0;j<bn;j++) b[j+bn]=(b[j]+arr[i])%m;
		sort(b,b+bn*2);
		bn=unique(b,b+bn*2)-b;
	}
	for (int i=0;i<an;i++)
	{
		int x=lower_bound(b,b+bn,m-a[i]-1)-b;
		if (b[x]!=m-a[i]-1) x--;
		ans=max(ans,a[i]+b[x]);
	}
	printf("%d\n",ans);
	return 0;
}

登录 *


loading captcha image...
(输入验证码)
or Ctrl+Enter