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. 【转】Python基础-字符串

    原文地址http://blog.chinaunix.net/uid-21169302-id-446256.html Python-String-Function 字符串中字符大小写的变换: * S.l ...

  2. Apache Flink - Component Stack

    作为一个软件堆栈,Flink是一个分层的系统.堆栈的不同层构建在彼此之上,并提高程序表示的抽象级别: 在runtime层以JobGraph的形式接受一个程序.JobGraph是一个通用的并行数据流,包 ...

  3. Java核心复习——CompletableFuture

    介绍 JDK1.8引入CompletableFuture类. 使用方法 public class CompletableFutureTest { private static ExecutorServ ...

  4. STL算法之find

    定义 template <class InputIterator, class T> InputIterator find (InputIterator first, InputItera ...

  5. 《Glibc内存管理》笔记DAY3

    目录 边界标记法 内容来源 边界标记法 /* conversion from malloc headers to user pointers, and back */ #define chunk2me ...

  6. arcgis python 参数类型和含义

    数据类型 datatype 关键字 描述 地址定位器 DEAddressLocator 用于地理编码的数据集,存储地址属性.关联的索引以及用于定义将地点的非空间描述转换为空间数据这一过程的规则. 地址 ...

  7. https://www.atlassian.com/git/tutorials/git-gc

    https://www.atlassian.com/git/tutorials/git-gc The git gc command is a repository maintenance comman ...

  8. 【Oracle/Java】批量删除16张十万数据的表 单线程耗时45秒 多线程耗时38秒

    昨天做了插入的单线程多线程比较,今天做个删除的. 单线程批量删除类代码: package com.hy.delete.singlethread; import java.sql.Connection; ...

  9. 有关Python的import...和from...import...的区别

    1.语法分析:首先from A import a1 是从A模块导入a1工具(可以是某个 函数,全局变量,类),import A是导入整个A模块的全部内容(包括全部的函数,全局变量,类). 2.内存分析 ...

  10. SVN分支创建与合并

    SVN分支 一个branch是某个development line(通常是主线也即trunk)的一个拷贝,branch存在的意义在于,在不干扰trunk的情况下,和trunk并行开发,待开发结束后合并 ...