【Ozon Tech Challenge 2020 F】Kuroni and the Punishment 题解
题目大意
有一个正整数序列 $a_1,\cdots,a_n$,每次操作可以把一个数 $+1$ 或 $-1$,但要使其仍为正数。问至少多少次操作,使得整个序列的 $\gcd$ 不为 $1$。
$n \le 2 \times 10^5$
2.5s
题解
首先有些很明显的观察,比如,可以枚举一个 $a_i$ 的质因数 $p$ 作为 $\gcd$,然后每个数变成 $\min(a_i \bmod p,p-a_i \bmod p)$,这样判一次是 $O(n)$ 的。
再比如,答案不会超过序列中奇数的数量,因为每个奇数 $+1$ 就会使得 $2|\gcd$。
然后。。。
就没有然后了。。。
然后题解说,虽然判一个质因子要 $O(n)$,但是只用判很少的质因子。
由观察 2,答案不超过 $n$,因此最终改动在 $1$ 以内的位置至少有 $\frac n2$ 个。也就是说,任选一个位置,至少有 $\frac n2$ 的概率,最终的 $\gcd$ 被它的质因子整除。
因此就随机选序列的 20 个数,判断 $a_i,a_i+1,a_i-1$ 的所有质因子,就行了~
代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50
| #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); }
|