杭电OJ [1005](http://acm.hdu.edu.cn/showproblem.php?pid=1005):

#####Problem Description
> A number sequence is defined as follows: f(1) = 1, f(2) = 1, f(n) = (A * f(n - 1) + B * f(n - 2)) mod 7. Given A, B, and n, you are to calculate the value of f(n). #####Input
> The input consists of multiple test cases. Each test case contains 3 integers A, B and n on a single line (1 <= A, B <= 1000, 1 <= n <= 100,000,000). Three zeros signal the end of input and this test case is not to be processed. #####Output
> For each test case, print the value of f(n) on a single line. #####Sample Input
> 1 1 3
1 2 10
0 0 0 #####Sample Output
> 2
5 题目对于那些ACM选手来说,肯定不是什么大问题,不过对于我们这种只能刷刷水题的人来说,还是有点困难的。 我看到题目之后的第一反应是对于每一个特定的A和B,每次的计算结果都存到数组里,这样不用每个输入都重新计算,可以节省一定量的时间,当时觉得这个想法已经不错了,但是还是TLE了。最后上网找了答案,发现很多答案里都提到f(n)的值其实是循环的,而且最大的循环长度不会超过49(即起码在n=49之前,f(n)的值就开始循环了,f(k)=f(1),f(k+1)=f(2),f(k+2)=f(3),...,k<=49)。但是网上很多文章并没有指出怎么才能发现,或者说推导出这个规律。最后我花了点时间,自己推了下才终于知道了发现规律的方法。 首先,观察到递推式里有mod 7,就知道f(n)的所有值都在[0, 6]之间,有7个取值。 然后,观察递推式,f(n) = (A * f(n - 1) + B * f(n - 2)) mod 7,A和B是定常数,而f(n - 1)和f(n - 2)的取值都分别有7种可能,也就是说f(n)的取值最多有49种组合(这49种组合中,和可能相同,但是代表的意义不同,例如1+4和2+3是不同的,2+3和3+2也是不同的)。当n > 51时(因为这种组合是从n=3开始算的),f(n)的取值组合必然是之前出现过了的,也就是必存在
```
f(n) = f(k), f(n - 1) + f(n - 2) = f(k - 1) + f(k - 2)
f(n - 1) = f(k - 1)
f(n - 2) = f(k - 2)
```
但是其实我们可以知道,f(n) = 0 + 0,这种情况是不可能的(因为这样话很容易证明对于所有的n,f(n)都是为0),所以其实最多只有48种情况,即从n = 50开始,组合就必然出现重复了。
其次,我们知道了f(n)和f(k)的取值组合完全相同后,只要证明f(n + 1) = f(k + 1)的即可证明f(n)的取值在n > 49后必然存在循环。
```
f(n + 1) = f(n) + f(n - 1)
f(n) = f(k)
f(n - 1) = f(k - 1)
```
由以上条件可知,f(n + 1) = f(k + 1),同理可以推导出f(n + 2) = f(k + 2),...,等等。并最终证明f(n)的值是循环的。 最后,我们已经证明了f(n)是存在循环的,最后要证明的是f(n)是整循环的,也就是说循环起始点应该是f(n) = f(n - 1) + f(n - 2) = f(1) + f(2)。先假设f(n + 2) = f(5) = f(n + 1) + f(n) = f(4) + f(3),n是循环开始点,由假设可以推导出f(n + 1) = f(n) + f(n - 1) = f(4) + f(3),f(n - 1)和f(3)之前存在两种可能:
1. f(n - 1) = f(3)
2. |f(n - 1) - f(3)| = 7
由f(n)的取值范围[0, 6]知,第二种情况是不可能的,所以f(n - 1) = f(3),所以n - 1是循环开始点,依次可以类推n - 2是循环开始点,...,直到f(n - k) = f(3) = f(n - k - 1) + f(n - k - 2) = f(2) + f(1),n - k - 2是循环开始点。所以由证明可知,如果n是循环开始点,则f(n) = f(1),f(n + 1) = f(2),...。 知道了f(n)的值是循环的之后,这道题目就很容易做了,只要求出循环开始点就行了,即f(i - 1) = 1, f(i) = 1。 具体实现代码如下:
#include <iostream>
using namespace std; int f[50] = {0, 1, 1};
int a, b, n;
int main() {
while (cin >> a >> b >> n) {
if (a == 0 && b == 0 && n == 0) {
break;
}
if (n > 2) {
int i;
for (i = 3; i <= 49; i++) {
f[i] = (a * f[i - 1] + b * f[i - 2]) % 7;
if (f[i] == f[2] && f[i - 1] == f[1]) {
break;
}
}
i -= 2;
n = n % i == 0 ? i : n % i;
}
cout << f[n] << endl;
}
}
 
												

从HD OJ 1005想到的的更多相关文章

  1. (light OJ 1005) Rooks dp

    http://www.lightoj.com/volume_showproblem.php?problem=1005        PDF (English) Statistics Forum Tim ...

  2. Light oj 1005 - Rooks (找规律)

    题目链接:http://www.lightoj.com/volume_showproblem.php?problem=1005 纸上画一下,找了一下规律,Ank*Cnk. //#pragma comm ...

  3. 九度OJ 1005:Graduate Admission (排序)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:5646 解决:1632 题目描述: It is said that in 2011, there are about 100 graduat ...

  4. 九度oj 1005

    题目1005:Graduate Admission 时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:6182 解决:1791 题目描述:                        It ...

  5. 百炼OJ - 1005 - I Think I Need a Houseboat

    题目链接:http://bailian.openjudge.cn/practice/1005/ 思路 一个半圆面积每年增长50,已知一个点坐标,求第几年点在半圆内. #include <stdi ...

  6. Light OJ 1005 - Rooks(DP)

    题目大意: 给你一个N和K要求确定有多少种放法,使得没有两个车在一条线上. N*N的矩阵, 有K个棋子. 题目分析: 我是用DP来写的,关于子结构的考虑是这样的. 假设第n*n的矩阵放k个棋子那么,这 ...

  7. Light Oj 1005

    题意: 从 n*n 的棋盘中放置 K 个 行和列不冲突的棋子 思路: 组合数学, 先选 k 个 行, k 个列, 就是 C(n,k) ^ 2; 然后 K 个棋子不相同, K ! 全排列 #includ ...

  8. Light OJ 1005 - Rooks 数学题解

    版权声明:本文作者靖心.靖空间地址:http://blog.csdn.net/kenden23/,未经本作者同意不得转载. https://blog.csdn.net/kenden23/article ...

  9. [杭电oj][1005]Number Sequence

    sky同学在努力地刷题..,在这题卡住了,于是一起研究了一下... 这题本身挺简单的,(1) = 1, f(2) = 1, f(n) = (A * f(n - 1) + B * f(n - 2)) m ...

随机推荐

  1. 浅谈HTTP中Get与Post的区别_转

    可参考:HTTP请求中POST与GET的区别 Http定义了与服务器交互的不同方法,最基本的方法有4种,分别是GET,POST,PUT,DELETE.URL全称是资源描述符,我们可以这样认为:一个UR ...

  2. select 5种子句介绍

    一.Where 条件查询 ①where expression 用法:expression为真,则该行取出 运用场合 各种条件查询场合,如按学号查学生,按价格查商品,按发布时间查新闻等 ②select ...

  3. asp.net 正在加载/处理(兼容IE Chrome)

    正在加载分两种 一种是页面初始化效果,也就是数据读取前,这个时候需要用ajax先显示读取中数据,当ajax回调的时候,读取页面数据 一种是界面读取中效果,就是用div显示,当OnPreRenderCo ...

  4. EntityFramework使用Code First模式创建数据库控制生成单数形式的表名

    使用Code-First模式生成数据库时,默认生成的数据库表的名称为类型的复数形式,例如实体类名称是"User",默认生成的数据库表名为“Users”,多数情况下我们并不想生成的数 ...

  5. mysql实现经纬度计算两个坐标之间的距离

    DELIMITER $$CREATE DEFINER = CURRENT_USER FUNCTION `getDistance`(`lon1` float,`lat1` float,`lon2` fl ...

  6. 数学 + 带权中位数 - SGU 114 Telecasting station

    Telecasting station Problem's Link Mean: 百慕大的每一座城市都坐落在一维直线上,这个国家的政府决定建造一个新的广播电视台. 经过了许多次试验后,百慕大的科学家们 ...

  7. 【转】伪O2O已死?2016年实体零售将迎来真正的O2O

    O2O果真如所谓的经济学家许小年所说“是两边都是零,中间一个2货”吗?我觉得,经济学家不是说相声的,这种哗众取宠的观点不应该出自一个严谨的经济学家之口.而且,我也不认为许小年真正懂什么叫O2O. 但O ...

  8. boost 互斥体和锁

    1.共享资源是一个自动锁住的房间,互斥体是钥匙,进入房间必须取钥匙,离开房间应该还钥匙.这就对应着互斥体的lock(取钥匙)和unlock(还钥匙). 2.考虑下面的场景:还钥匙的时候出现异常,会发生 ...

  9. C语言若干知识点归记

    一.C语言指针学习架构 1.基本数据类型---指针 2.字符串---指针 3.数组---指针 4.函数---指针 5.结构体---指针 6.共用体---指针 7.枚举---指针 8.位域---指针 9 ...

  10. SVN入门 服务器VisualSVN Server和客户端TortoiseSVN安装

    Subversion是一个版本控制系统,相对于的RCS.CVS,采用了分支管理系统,它的设计目标就是取代CVS.互联网上免费的版本控制服务多基于Subversion. 一.SVN工作原理 SVN(Su ...