目录
(一)问题描述
(二)思路分析
(三)代码 + 注释
(四)总结
一:
【问题描述】
数学老师给小明出了一道等差数列求和的题目。但是粗心的小明忘记了一部分的数列,只记得其中 N 个整数。
现在给出这 N 个整数,小明想知道包含这 N 个整数的最短的等差数列有几项?
【输入格式】
输入的第一行包含一个整数 N。
第二行包含 N 个整数 A1 ,A2 ,··· ,AN 。(注意 A1 ∼ AN 并不一定是按等差数列中的顺序给出)
【输出格式】
输出一个整数表示答案。
【样例输入】
5
2 6 4 10 20
【样例输出】
10
【样例说明】
包含 2、6、4、10、20 的最短的等差数列是 2、4、6、8、10、12、14、16、18、20。
【评测用例规模与约定】
对于所有评测用例,2 ≤ N ≤ 105,0 ≤ Ai ≤ 109 。
二:
思路分析:
- 储备知识:
(1) 等差数列的求n项的公式:n = (an-a1) / d + 1;
(2) 等差数列的每一项利用公差表达的通式:(下标从1开始)a1,a2,a3…an。
a2 - a1 = d,
a3 - a1 = 2d,
…
an - a1 = (n-1)d;
所以每一项与第一项的差值都是公差d的倍数。d是差值的公约数!
(3) 求最大公约数的代码:
int gcd (int a, int b)
{
return b ? gcd (b, a % b) : a;
}
2.从上面求n的公式中可以看出,要令 n 尽可能小那么分子要尽可能大,分母要尽可能小,所以从分子分母角度分析:
ps:求的是包含这些项最短的等差数列,而不是完整的等差数列!
3.分母:因为要求的是这个数列的项数,所以尽可能是最大值 - 最小值(将输入的最大的数视作最大值,输入的最小的数视作最小值,构造数列的左右端点固定),然后根据公式就可求出端点内(包括端点)的项数。
举例:
2 4 6 8 10 12
a1 a2 a3 a4 a5 a6;
若输入的残缺数列是 : 4 8 10;然后补充这个数列。d = 2.那么首先知道这个数列的项数 n = (a5 - a2 ) / d + 1 = 4项,所以从 4 6 8 10自动构成了一个新的数列。这就是为啥用残缺数列的最大值 - 最小值呢?
4.分子:分子是公差,是每一项与第一项差值的公约数,若令d最大即求这些差值的最大公约数,将这个最大公约数视作公差!
5.最后记得特判一下数列公差 = 0的情况,因为当公差等于0时,此时的数列每个数都相等,又要求n尽可能小,那么此时n就是初始输入的n。就是原数列!
三:
代码:
#include<bits/stdc++.h>
using namespace std;
const int N = 1e5 + 10;
int a[N], x[N];
int gcd (int a, int b)
{
return b ? gcd (b, a % b) : a;
}
int main ()
{
int n;
cin >> n;
for (int i=0; i < n; i ++)
{
cin >> a[i];
}
sort (a, a + n);
//存储每一项与第一项的差值!
for (int i=1; i < n; i ++)
{
x[i] = a[i] - a[i-1]; //i + 1 < n防止越界的情况!
}
int d = 0;
for (int i=1; i < n; i ++)
{
d = gcd (d, x[i]);
}
if (d == 0) cout << n << endl; //公差 = 0的情况!
else cout << (a[n-1] - a[0]) / d + 1 << endl;
return 0;
}
/*
思路:
1.
*/