1283: Hamster

参考博客

wust_tanyao的博客

题目

  第0个月有1对仓鼠。仓鼠的寿命是M个月,仓鼠成年后每个月生一对仓鼠(一雌一雄),问N个月后有仓鼠多少对。更多内容点此链接

题目分析

  阅读之前先看看两组数据,如果能推出结果,可以跳过题目分析部分。

  输入:

3 4
5 2

  输出:

3
4

  OK,如果你推算不出来,可以参考下面这个表的推算过程:

  代码中只用了第一个参数(birth[])和第三个参数(adult)。其他参数可直接算出来。递推语句是:

birth[i]等于第i-1个月的adult

i个月的成年数量adult等于第i-1个月的成年数量和出生数量之和减去M+1个月之前的那个月的出生数量(i<M+1就不用减)。

最后结果是第i个月的成年数量加出生数量。

—————————————————————————————————————————

第二条中的减去的那部分,其实是这个月死去的仓鼠数量。因为那一个月出生的仓鼠刚好在这个月死亡。


  如果我说的不是很清楚的话,可以自己对照表格自己慢慢推敲,多思考,多思考,多思考。不要放弃,真滴不难。

  每次碰到这类题,我都要想半天,理不清楚思路QAQ……下面欣赏代码吧&_&

代码

/**
* time 447ms
* @author PengHao
* @version A3.2
* @date 2019-04-26 上午10:23:03
*/ import java.io.BufferedInputStream;
import java.util.Scanner; public class Main { public static void main(String[] args) {
Scanner sc = new Scanner(new BufferedInputStream(System.in));
long[] birth = new long[10001]; // 每个月出生的仓鼠(对)
int mod = 1000000007; // 取余运算的模
int N, M; // N个月后存活多少对仓鼠,仓鼠的寿命为M个月
int M1; // M1=M+1,循环里面要用到,提前算出来节约时间
birth[0] = 1; // 第0个月买了一对刚“出生”的仓鼠,所以为1
long adult; // 某个月成年的仓鼠数量(对)
while (sc.hasNext()) {
N = sc.nextInt();
M = sc.nextInt();
M1 = M + 1;
adult = 0; // 第0个月成年仓鼠为0对
// 循环体内计算过i个月后的出生仓鼠数量birth[i]和成年仓鼠数量adult
for (int i = 1; i <= N; i++) {
// 上个月成年仓鼠数量(成年后每个月生一对),就是这个月出生仓鼠数量
birth[i] = adult;
// 这个月成年仓鼠数量是上个月成年仓鼠数量和上个月出生仓鼠数量之和
// 减去上个月死亡的仓鼠数量(死的是M+1个月前的那个月出生的仓鼠)
// 如果没有死的就不用减
adult += birth[i - 1];
if (i > M) {
adult -= birth[i - M1];
}
// 前面死的仓鼠数量(birth[i-M1])可能比后面出生的仓鼠(birth[i-1])
// 多很多,if语句里面就有可能减成负的。
adult = (adult + mod) % mod; // 因此需要取余
}
// 这个月成年数量加出生数量就是这个月存活数量。
// 这里的成年数量是已经减去了这个月死亡数量的(if语句的作用)。
System.out.println((adult + birth[N]) % mod); // 可能过大,要取余
}
sc.close();
} }
代码补充

  思路理清楚了,代码其实不难。需要注意的地方是:1、哪些地方取余,参考博客的代码中取余有重复的,我将它删除掉了;2、int型不够,要用long型;3、参考博客的数组的第0个值我拿到了最外面,减少重复计算;4、将“指针”结构(head,tail)去掉,直接用i操作,减少运算缩短时间;5、循环中要用到M+1,拿到for循环外面定义为M1,减少重复计算。

  这算是斐波拉契的一种变形吧,以前做的那种你可以看成寿命是无穷的,现在只不过多了限制条件而已,减掉那些死亡的就可以了。

  这类找规律的题(从表格就可以看出来),一般都是可以预处理的,但是这道题目预处理的代价太大(内存开销有好几百MB),申请内存时候就挂了。所以我就没用预处理。如果大家想到了优化的预处理方法,欢迎告知本人,谢谢。


写在最后:

  1. 如需转载,请于首页至少注明链接形式的wowpH的博客这几个字;
  2. 代码原创,如需公开引用,不能删除首行注释(作者,版本号,时间等信息)。
  3. 如果有疑问欢迎评论留言,尽量解答。

WUSTOJ 1283: Hamster(Java)的更多相关文章

  1. WUSTOJ 1285: Factors(Java)

    1285: Factors 参考   hadis_fukan的博客--wustoj 1285 Factors 题目   输入一个数n,找出1~n之间(包括1,n)的质因子最多的数(x)的质因子个数(f ...

  2. WUSTOJ 1319: 球(Java)并查集

    题目链接:1319: 球 参考:wustoj 1319 球-wust_tanyao,并查集 并查集系列:WUSTOJ 1346: DARK SOULS(Java)并查集 Description Icy ...

  3. WUSTOJ 1326: Graph(Java)费马数

    题目链接:1326: Graph 参考博客:HNUSTOJ-1617 Graph(费马数)--G2MI Description Your task is to judge whether a regu ...

  4. WUSTOJ 1349: TLE(Java)算法优化

    题目链接:1349: TLE Description WH在刷题时,设计出了如下代码: #include<stdio.h> int main() { int i, j, cnt, k, N ...

  5. WUSTOJ 1282: Start(Java)

    1282: Start 题目   判断一个字符串是不是回文串.例如:"abcba"是回文串.更多内容点击标题. 分析   水题,自己思考. 代码 /** * time 838ms ...

  6. WUSTOJ 1347: GCD(Java)互质

    题目链接:1347: GCD Description 已知gcd(a,b)表示a,b的最大公约数. 现在给你一个整数n,你的任务是在区间[1,n)里面找到一个最大的x,使得gcd(x,n)等于1. I ...

  7. WUSTOJ 1325: Distance(Java)坐标计算

    题目链接:1325: Distance Description There is a battle field. It is a square with the side length 100 mil ...

  8. WUSTOJ 1313: 数列(Java)进制转换

    题目链接:

  9. WUSTOJ 1308: 采药(Java)动态规划-01背包

    题目链接:

随机推荐

  1. kvm 学习(二)镜像

    Linux下 如何通过命令行使用现有的镜像创建.启动kvm虚拟机 这里假定已经创建好了相应的镜像: eg:我这里制作的镜像名称为zu1-centos7.img # ls zu1-centos7.img ...

  2. C语言中的多线程

    原文:https://www.cnblogs.com/yorkyang/p/7804733.html 线程的优点: 减少系统调度开销,不占有独立的资源,切换速度快,执行效率高. 线程间通信方便,可共享 ...

  3. wireshark新手导航

    本篇教你如何开始使用Wireshark.开始本篇之前,请先已经安装完wireshark 3.0.0. 初始界面 启动wireshark,可以看到程序界面主要由以下几部分组成: 窗口标题-window ...

  4. FCRA题库及答案(未完待续)

    一.FCRA-入门基础(23) 1.FineReport中普通模板总共有5种不同的预览模式:分页预览.填报预览.新填报预览.数据分析.移动端预览 决策报表下为表单预览.移动端预览 2.FineRepo ...

  5. yarn集群8088端口被攻击挖矿

    yarn集群8088端口被攻击挖矿 https://www.jianshu.com/p/a2d6b0d827ab https://blog.csdn.net/lm709409753/article/d ...

  6. 打印变量地址-0x%08x

    地址是8个16进制数. 1.8个16进制数:相当于32个二进制数.4G内存刚好可以用32位的二进制表示出来.2.因为变量或函数等等在运行时都是存储在内存中的,所以你用取地址符当然是取出计算机内存中的地 ...

  7. 自定义控件之canvas变换和裁剪

    1.平移 //构造两个画笔,一个红色,一个绿色 Paint paint_green = generatePaint(Color.GREEN, Paint.Style.STROKE, 3); Paint ...

  8. 【集成模型】Boosting

    0 - 思想 Bagging算法思想是减少预测方差(variance),Boosting算法思想是为了减少预测偏差(bias). Boosting算法思想是将“弱学习算法”提升为“强学习算法”.一般来 ...

  9. sha256C代码例子

    a.c #include <stdio.h>#include <string.h>#include <openssl/sha.h> 1 int main(int a ...

  10. rank SQL 筛选重复数据

    先思考一个问题: 看下面的表数据 问题:现在需要在 A 和 B 相同的前提下对 C desc排序,然后拿到排序中不是第一个的数据?也就是说拿到下面的数据 只用一条 SQL 实现: select * f ...