【Ozon Tech Challenge 2020 F】Kuroni and the Punishment 题解

题目大意

  有一个正整数序列 a 1 , ⋯   , a n a_1,\cdots,a_n a1,,an,每次操作可以把一个数 + 1 +1 +1 − 1 -1 1,但要使其仍为正数。问至少多少次操作,使得整个序列的 gcd ⁡ \gcd gcd 不为 1 1 1
   n ≤ 2 × 1 0 5 n \le 2 \times 10^5 n2×105
  2.5s

\\
\\
\\

题解

  首先有些很明显的观察,比如,可以枚举一个 a i a_i ai 的质因数 p p p 作为 gcd ⁡ \gcd gcd,然后每个数变成 min ⁡ ( a i   m o d   p , p − a i   m o d   p ) \min(a_i \bmod p,p-a_i \bmod p) min(aimodp,paimodp),这样判一次是 O ( n ) O(n) O(n) 的。
  再比如,答案不会超过序列中奇数的数量,因为每个奇数 + 1 +1 +1 就会使得 2 ∣ gcd ⁡ 2|\gcd 2gcd
  然后。。。
  就没有然后了。。。

  然后题解说,虽然判一个质因子要 O ( n ) O(n) O(n),但是只用判很少的质因子。
  由观察 2,答案不超过 n n n,因此最终改动在 1 1 1 以内的位置至少有 n 2 \frac n2 2n 个。也就是说,任选一个位置,至少有 n 2 \frac n2 2n 的概率,最终的 gcd ⁡ \gcd gcd 被它的质因子整除。
  因此就随机选序列的 20 个数,判断 a i , a i + 1 , a i − 1 a_i,a_i+1,a_i-1 ai,ai+1,ai1 的所有质因子,就行了~

代码

#include<bits/stdc++.h>
#define fo(i,a,b) for(int i=a;i<=b;i++)
using namespace std;

typedef long long LL;

const int maxn=2e5+5;

int n;
LL a[maxn];

LL _check(LL p)
{
	LL ans=0;
	fo(i,1,n) ans+=(a[i]<p) ?p-a[i] :min(a[i]%p,p-a[i]%p);
	return ans;
}
LL check(LL x)
{
	LL ans=n+500;
	int sqrtx=sqrt(x);
	for(int i=2; i<=sqrtx; i++) if (x%i==0)
	{
		ans=min(ans,_check(i));
		while (x%i==0) x/=i;
	}
	if (x>1) ans=min(ans,_check(x));
	return ans;
}

int main()
{
	scanf("%d",&n);
	fo(i,1,n) scanf("%lld",&a[i]);
	
	LL ans=0;
	fo(i,1,n) ans+=(a[i]&1);
	
	random_shuffle(a+1,a+1+n);
	random_shuffle(a+1,a+1+n);
	random_shuffle(a+1,a+1+n);
	fo(i,1,min(n,20))
	{
		ans=min(ans,check(a[i]-1));
		ans=min(ans,check(a[i]));
		ans=min(ans,check(a[i]+1));
	}
	
	printf("%lld\n",ans);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值