CF926B Add Points

一道尚未评定的水题

更好的阅读体验

思路

来分析分析样例:

3
-5 10 5


我们把它升序排列,会得到这个东西↑

仔细地观察后可以发现:加一个(0,0)的点显然是最优的

用脚趾头想想为什么,我们发现,这题题意就是想让我们把一段线段x等分,使得给出的点都是它的x等分点。

而通过我们的敏锐的第六感做题经验,不难看出最优解即相邻点距离的最大公约数

做法

显然此题难点在于求n个数的gcd

而要求多个数的gcd,两两求之后合并即可。

证明:

以三个数为例,设有a,b,c三数,x=gcd(a,b),y=gcd(x,c),因为a%x=0,x%y=0,所以a%y也等于零,且y是符合条件的最大值。

所以我们可以递推求n个长度的gcd,然后出解即可

代码

#include <iostream>
#include <cmath>
#include <algorithm>
using namespace std;
int a[100001];
int b[100001];
int main()
{
int n;
cin >> n;
for ( int i = 1; i <= n; i++ )
{
cin >> b[i];
}
sort ( b + 1, b + n + 1 );
for ( int i = 2; i <= n; i++ )
a[i] = b[i] - b[i-1];
int dis = b[n] - b[1];
for ( int i = 2; i <= n; i++ )
{
a[i] = __gcd( a[i], a[i-1] );
//在<cmath>里的神奇函数(妈妈再也不用担心我不会辗转相除啦!)
}
cout << dis / a[n] - n + 1;
/*此时的a[n]即为最优的线段长度,dis/a[n]为线段个数,
再加1是点的个数,最后减去原有的n个点,输出*/
return 0;
}

完美收官(你们没发现开头的链接是两个吗

posted @ 2019-08-12 16:56  _Au  阅读(270)  评论(0编辑  收藏  举报