本解析禁止用于商业用途
禁止在未经许可的情况下转载
本篇仅在CSDN发布
官方解析请访问24年东方博宜OJ编程月赛评讲课 - 网易云课堂
题目描述
炎热的夏天来了,小 A 买了半个西瓜,打算和信奥班的同学们一起分享。
老师借给了小 A 一把锋利的西瓜刀,小 A 可以借助这把西瓜刀,每次在西瓜的圆形切面上,完整且不弯曲的切一刀。
假设西瓜无限大,因此可以在这个圆形切面上无限切,请问小 A 切了 n 刀后,可以最多得到多少片西瓜。
下图给出了小 A 同学分别切 1 刀、 2 刀、 3 刀可以得到最多西瓜数的切法。
输入
一行一个整数 n ,表示切西瓜的次数。
输出
一行一个整数,表示最多能切出多少片西瓜。
样例
输入
2
输出
4
输入
7
输出
29
输入
30
输出
466
说明
数据范围
对于 30% 的数据,n≤3。
对于另外 40% 的数据, n≤10^5 。
对于另外 30% 的数据, n≤10^9 。
要解决这个问题,我们需要理解每次切割如何影响西瓜片的数量。每次切割时,我们希望尽可能多地增加西瓜片的数量。通过观察和推理,我们可以发现一个规律:每次切割时,如果新的切割线与之前的所有切割线相交,那么可以最大化西瓜片的数量。
具体来说,切第 n 刀时,如果这刀与之前的 n−1 刀都相交,那么会增加 n 片西瓜。因此,切 n 刀后,最多可以得到的西瓜片数是前 n 个自然数的和,即:
Sn=1+2+3+⋯+n=n(n+1)/2+1
这个公式可以通过数学归纳法证明。
具体步骤
- 输入处理:读取输入的整数 n。
- 计算结果:使用公式 n(n+1)/2+1 计算最多能切出的西瓜片数。
- 输出结果:将计算结果输出。
代码实现
#include <iostream>
int main() {
long long n;
std::cin >> n;
// 计算最多能切出的西瓜片数
long long result = (n * (n + 1)) / 2 + 1;
// 输出结果
std::cout << result << std::endl;
return 0;
}
解释
- 输入处理:使用
std::cin
读取输入的整数 n。 - 计算结果:使用公式 n(n+1)/2+1 计算最多能切出的西瓜片数。这里使用
long long
类型来避免整数溢出问题,因为 n 可能非常大(达到 10^9)。 - 输出结果:使用
std::cout
输出计算结果。
这个算法的时间复杂度是 O(1),因为计算公式直接给出了结果,不需要循环或递归。因此,它可以高效地处理大规模数据。