动态规划之Fib数列类问题应用

一,问题描述

有个小孩上楼梯,共有N阶楼梯,小孩一次可以上1阶,2阶或者3阶。走到N阶楼梯,一共有多少种走法?

二,问题分析

DP之自顶向下分析方式:

爬到第N阶楼梯,一共只有三种情况(全划分,加法原理),从第N-1阶爬1阶到第N阶;从第N-2阶爬2阶到第N阶;从第N-3爬3阶到第N阶

故:way(N)=way(N-1)+way(N-2)+way(N-3)

这与求Fib数列非常相似,当然,其他类似的问题也可以这样求解。

初始条件:

way(1)=1

way(2)=2

way(3)=4

三,代码实现

public class WaysOfLadder {

    public static int ways(int n){
        if(n <= 0)
            throw new IllegalArgumentException();
        return waysLadder(n);
    }

    //递归算法爬上第n阶楼梯一共需要多少种方式
    private static int waysLadder(int n){
        assert n > 0;
        //base condition
        if(n == 1)
            return 1;
        if(n == 2)
            return 2;
        if(n == 3)
            return 4;
        else
            return waysLadder(n-1) + waysLadder(n - 2) + waysLadder(n - 3);
    }

    //dp
    public static int ways_dp(int n){
        if(n <= 0)
            throw new IllegalArgumentException();

        int pre_1 = 1;
        int pre_2 = 2;
        int pre_3 = 4;
        int res = 0;
        for(int i = 4; i <= n; i++)
        {
            res = pre_1 + pre_2 + pre_3;
            pre_1 = pre_2;
            pre_2 = pre_3;
            pre_3 = res;
        }
        return res;
    }

    public static void main(String[] args) {
        int n = 32;
        System.out.println(ways_dp(n));
        System.out.println(ways(n));
    }
}

上面代码清晰地对比了DP实现与递归实现的方式。DP是用三个变量保存当前计算的结果,当计算下一个结果时,先“查表”再计算。而递归则是使用三个递归函数调用,递归函数调用计算了大量的重叠的子问题,每次递归调用都要压栈、出栈。递归的时间复杂度为O(3^N),而DP的时间复杂度为O(N)

类似的思想,还有计算杨辉三角的公式:C(n,r)=C(n-1,r) + C(n-1,r-1)具体可参考

只不过杨辉三角的计算公式有两个参数而已。

另外,相关问题可参考:组合问题与动态规划的联系之应用

时间: 2024-10-25 03:41:42

动态规划之Fib数列类问题应用的相关文章

acm 之fib数列——java

1022. Fib数列 Description 定义Fib数列:1,1,2,3,5,8,13,… 求第N项除以2010的余数 Input Format 输入仅一行,为一个整数N Output Format 输出仅一行,为第N项除以2010的余数 Sample Input 3 Sample Output 2 Limits: 对于70%的数据 N≤1,000,000 对于100%的数据 N≤210,000,000,000 这道题最让人着急的是数太大了,而且不可以使用递归,这种方法会暴栈,但是java

HDU 1005 Number Sequence 矩阵乘法 Fib数列

原题: http://acm.hdu.edu.cn/showproblem.php?pid=1005 题目大意: 按规律求出第n项. 由矩阵乘法我们可以知道: 所以对于fib数列我们可以用矩阵来求,由于矩阵可以左乘右乘,所以我们可以用快速幂来优化. #include<iostream> #include"string.h" #include<stdio.h> using namespace std; const int bc=2; const int mod =

FIB数列

斐波那契级数除以N会出现循环,此周期称为皮萨诺周期. 下面给出证明 必然会出现循环 这是基于下面事实: 1. R(n+2)=F(n+2) mod P=(F(n+1)+F(n)) mod P=(F(n+1) mod p +F(n) modp) mod p 2. 斐波那契数列的最大公约数定理:gcd(F(m),F(n))=F(gcd(m,n)) 最大公约数定理表明如果F(k)能被N整除,则F(ik)也能被N整除,这就表明了斐波那契数列所含因子的周期性,下面列举: 因子:2,3,4,5, 6,7,8,

1022. Fib数列

Description 定义Fib数列:1,1,2,3,5,8,13,- 求第N 项除以2010 的余数 Input Format 输入仅一行,为一个整数N Output Format 输出仅一行,为第N 项除以2010 的余数 Sample Input 3 Sample Output 2 Limits: 对于70%的数据 N≤1,000,000 对于100%的数据 #include<iostream>using namespace std; int main(){    int t;    

UVaLive 3357 Pinary (Fib数列+递归)

题意:求第 k 个不含前导 0 和连续 1 的二进制串. 析:1,10,100,101,1000,...很容易发现长度为 i 的二进制串的个数正好就是Fib数列的第 i 个数,因为第 i 个也有子问题,其子问题也就是Fib,这样就可以用递归来解决了. 代码如下: #pragma comment(linker, "/STACK:1024000000,1024000000") #include <cstdio> #include <string> #include

[bzoj5118]Fib数列2_费马小定理_矩阵乘法

Fib数列2 bzoj-5118 题目大意:求Fib($2^n$). 注释:$1\le n\le 10^{15}$. 想法:开始一看觉得一定是道神题,多好的题面啊?结果...妈的,模数是质数,费马小定理就tm完事了,将fib数列的通项公式列出来然后费马小定理... 最后,附上丑陋的代码... ...(照着郭爷一顿瞎jb敲) #include <iostream> #include <cstdio> #include <cstring> #include <algo

Fib数列问题(项数很大)

用fib(n)表示斐波那契数列的第n项,现在要求你求fib(n) mod m.fib(1)= 1, fib(2)= 1. 输入格式 输入2个整数n(1≤n≤1018), m(2≤m≤10000000). 输出格式 输出fib(n)对m取模的值. 样例输入1 4 10 样例输出1 3 样例输入2 100000000 100000000 样例输出2 60546875 1 #include<bits/stdc++.h> 2 using namespace std; 3 typedef long lo

fib数列变种题目

对一个正整数作如下操作:如果是偶数则除以2,如果是奇数则加1,如此进行直到1时操作停止,求经过9次操作变为1的数有多少个? 第9次操作:结果1由2产生.1个被操作数8:结果2只能由4产生.1个被操作数7:结果4由8.3产生.2个6:结果8由16.7产生:结果3由6产生.共3个5:结果16由32.15产生:结果7由14产生:结果6由12.5产生.共5个…每次操作,偶数(2除外)都由该数减1和该数的2倍得来,奇数只由该数的2倍得来各次操作的操作对象个数为:1,1,2,3,5,8,13,21,34,…

bzoj5118: Fib数列2(费马小定理+矩阵快速幂)

题目大意:求fib(2^n) 就是求fib矩阵的(2^n)次方%p,p是质数,根据费马小定理有 于是这题就完了 注意因为模数比较大会爆LL,得写快速乘法... #include<bits/stdc++.h> #define ll long long #define MOD(x) ((x)>=mod?(x-mod):(x)) using namespace std; const int maxn=500010; const ll mod=1125899839733759; struct m