#蓝桥杯————等差数列

目录


(一)问题描述
(二)思路分析
(三)代码 + 注释
(四)总结

一:
【问题描述】

数学老师给小明出了一道等差数列求和的题目。但是粗心的小明忘记了一部分的数列,只记得其中 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. 储备知识:
    (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. 
*/
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值