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. CF786E ALT

    题意 有一棵 \(n\) 个点的树和 \(m\) 个人,第 \(i\) 个人从 \(u_i\) 走到 \(v_i\) 现在要发宠物,要求一个人要么他自己发到宠物,要么他走的路径上的都有宠物. 求最小代 ...

  2. idea JRebel

    JRebel 链接:https://pan.baidu.com/s/11LI0RkPtrfEWQENns6cWAA 提取码:ndsu settings -> plugins -> inst ...

  3. cross socket和msgpack的数据序列和还原

    cross socket和msgpack的数据序列和还原 procedure TForm1.Button1Click(Sender: TObject); begin var pack: TSimple ...

  4. delphi 解决android 9上无法使用http协议

    delphi 解决android 9上无法使用http协议 安卓9不让客户端通过非https方式访问服务端数据(不允许发送明文http请求)的问题. 解决方法: 1.选择安卓平台编译一次程序,在项目根 ...

  5. linux设置su和sudo为不需要密码

    一 设置sudo为不需要密码   有时候我们只需要执行一条root权限的命令也要su到root,是不是有些不方便?这时可以用sudo代替.默认新建的用户不在sudo组,需要编辑/etc/sudoers ...

  6. matlab fspecial 用法解释

    Matlab 的fspecial函数用法 fspecial函数用于建立预定义的滤波算子,其语法格式为:h = fspecial(type)h = fspecial(type,para)其中type指定 ...

  7. OpenGL ES: (1) OpenGL ES的由来 (转)

    1. 电脑是做什么用的? 电脑又被称为计算机,那么最重要的工作就是计算.看过三体的同学都知道, 电脑中有无数纳米级别的计算单元,通过 0 和 1 的转换,完成加减乘除的操作. 2. 是什么使电脑工作? ...

  8. 自定义基于IFC数据的施工进度数据结构

    <DataSource>D:/qlbz20190802.ifc</DataSource> <Datas> <Data> <ID></I ...

  9. spark "main" java.lang.ArrayIndexOutOfBoundsException: 10582

    升级 你的 paranamer 到2.8 ,这是由于你的jdk版本1.8导致 <!-- https://mvnrepository.com/artifact/com.thoughtworks.p ...

  10. Delphi下Treeview控件基于节点编号的访问1

    有时我们需要保存和重建treeview控件,本文提供一种方法,通过以树结构节点的编号访问树结构,该控件主要提供的方法如下:      function GetGlobeNumCode(inNode:T ...