本文主要是介绍洛谷 P2669 [NOIP2015 普及组] 金币,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
本文由Jzwalliser原创,发布在CSDN平台上,遵循CC 4.0 BY-SA协议。
因此,若需转载/引用本文,请注明作者并附原文链接,且禁止删除/修改本段文字。
违者必究,谢谢配合。
个人主页:blog.csdn.net/jzwalliser
题目
洛谷 P2669 [NOIP2015 普及组] 金币
[NOIP2015 普及组] 金币
题目背景
NOIP2015 普及组 T1
题目描述
国王将金币作为工资,发放给忠诚的骑士。第一天,骑士收到一枚金币;之后两天(第二天和第三天),每天收到两枚金币;之后三天(第四、五、六天),每天收到三枚金币;之后四天(第七、八、九、十天),每天收到四枚金币……;这种工资发放模式会一直这样延续下去:当连续 n n n 天每天收到 n n n 枚金币后,骑士会在之后的连续 n + 1 n+1 n+1 天里,每天收到 n + 1 n+1 n+1 枚金币。
请计算在前 k k k 天里,骑士一共获得了多少金币。
输入格式
一个正整数 k k k,表示发放金币的天数。
输出格式
一个正整数,即骑士收到的金币数。
样例 #1
样例输入 #1
6
样例输出 #1
14
样例 #2
样例输入 #2
1000
样例输出 #2
29820
提示
【样例 1 说明】
骑士第一天收到一枚金币;第二天和第三天,每天收到两枚金币;第四、五、六天,每天收到三枚金币。因此一共收到 1 + 2 + 2 + 3 + 3 + 3 = 14 1+2+2+3+3+3=14 1+2+2+3+3+3=14 枚金币。
对于 100 % 100\% 100% 的数据, 1 ≤ k ≤ 1 0 4 1\le k\le 10^4 1≤k≤104。
想法
可以先观察一下这个数列:
1 , 2 , 2 , 3 , 3 , 3 , 4 , 4 , 4 , 4 , 5 , 5 , 5 , 5 , 5 , 6 , 6 , 6 , … 1,2,2,3,3,3,4,4,4,4,5,5,5,5,5,6,6,6,\dots 1,2,2,3,3,3,4,4,4,4,5,5,5,5,5,6,6,6,…
能得出来什么规律呢?稍稍推理一下,我们就可以得到以下公式:
假设第 n n n天会获取 m m m个金币,则
m n = 1 + ⌊ 8 n − 7 ⌋ 2 m_n=\frac{1+\lfloor \sqrt{8n - 7} \rfloor}{2} mn=21+⌊8n−7⌋
那么,总的金币个数为 ∑ i = 1 n m i \sum_{i=1}^n m_i i=1∑nmi
得出规律后,时间复杂度被压缩到了 O ( n ) O(n) O(n),不至于超时了。
至于推理的过程,如果不懂可以看这里:由1,2,2,3,3,3,4,4,4,4……推出公式
实现
- 用刚才得出的规律,写一个函数。
- 算出每一天拿到的金币数,将它们累加起来。
- 最后别忘记输出啊。
题解
C++
#include<bits/stdc++.h>
using namespace std;int coin(int n){ //计算第n天拿到的金币数return (1 + sqrt(8 * n - 7)) / 2; //套公式
}int main(){int n,total = 0;cin >> n; //输入for(int i = 1;i <= n;i++){total += coin(i); //调用函数,累加金币个数}cout << total; //输出return 0;
}
Python
import math
def coin(n): #计算第n天拿到的金币数return int((1 + int(math.sqrt(8 * n - 7))) / 2) #套公式total = 0
n = int(input()) #输入
for i in range(1,n + 1):total += coin(i) #调用函数,累加金币个数print(total) #输出
难度
难度:★★☆☆☆
这道题主要是推公式有点费解。公式推出来之后,题目迎刃而解,而且代码简洁。
结尾
你是怎么想的?欢迎留言啊!我们下期再见!(˵¯͒〰¯͒˵)
这篇关于洛谷 P2669 [NOIP2015 普及组] 金币的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!