当前位置:编程学习 > C/C++ >>

fibonacci数列和跳台阶问题

首先百度之后知道,fibonacci数列其实在科学中有着广泛的应用,下面一句话引子百度百科:在现代物理、准晶体结构、化学等领域,斐波纳契数列都有直接的应用。
了解fibonacci数列的人都知道它的推到公式:
   0                       n=0
f(n)=1                       n=1
         f(n-1)+f(n-2)   n>=2
有时候遇到的问题是如何求解fibonacci数列的各项的值,我记得有两种方法,就是递归方法和应用循环的方法。下面给出两种程序的代码:
[cpp] 
#include<stdio.h> 
int fibonacci_recursion(int n) 
{//递归算法 
    int result[2]={0,1}; 
    if(n<2) 
        return result[n]; 
    else 
        return fibonacci_recursion(n-1)+fibonacci_recursion(n-2); 

int fibonacci_non_recursion(int n) 
{//非递归算法 
    int result[2]={0,1}; 
    if(n<2) 
        return result[n]; 
    int i; 
    int fa=0,fb=1,fc; 
    for(i=1;i<n;i++) 
    { 
        fc=fa+fb; 
        fa=fb; 
        fb=fc; 
    } 
    return fc; 

void main() 

    int n; 
    printf("input n value:"); 
    scanf("%d",&n); 
    printf("%d\n",fibonacci_recursion(n)); 
    printf("%d\n",fibonacci_non_recursion(n)); 

上面的两个方法都是在O(N)的时间内得到解,但是在csdn的论坛上看到有人说,可以再O(logn)时间内求得fibonacci数列的第n项值,于是google了一下,找到了一篇博文,的确可以在O(logn)内求解。
有一个数学公式:{f(n), f(n-1), f(n-1), f(n-2)} ={1, 1, 1,0}n-1。
(注:{f(n+1), f(n), f(n), f(n-1)}表示一个矩阵。在矩阵中第一行第一列是f(n+1),第一行第二列是f(n),第二行第一列是f(n),第二行第二列是f(n-1)。)
下面就来证明一下这个公式的正确性:fibonacci的第0项为0,第一项和第二项分别是1,所以有公式{1,1,1,0},然后第三项为2,那么有公式{2,1,1,1},此时k=3,则计算第一个公式的平方结果是{2,1,1,1},结果正对,假设k=n-1时,公式成立,就有{f(n-1), f(n-2), f(n-2), f(n-3)} ={1, 1, 1,0}n-2
由fibonacci数列的递推式可以得到{f(n),f(n-1),f(n-1),f(n-2)}={f(n-1)+f(n-2),f(n-1),f(n-2)+f(n-3),f(n-2)},因为计算的原因,公式中第二项和第四项不用代换,可以看出,这个结果就是{f(n-1), f(n-2), f(n-2), f(n-3)}与{1,1,1,0}相乘的结果。故原式得证。
如果简单第从0开始循环,n次方将需要n次运算,并不比前面的方法要快。但我们可以考虑乘方的如下性质:
        /  an/2*an/2                      n为偶数时
an=
        \  a(n-1)/2*a(n-1)/2            n为奇数时
要求得n次方,我们先求得n/2次方,再把n/2的结果平方一下。如果把求n次方的问题看成一个大问题,把求n/2看成一个较小的问题。这种把大问题分解成一个或多个小问题的思路我们称之为分治法。这样求n次方就只需要logn次运算了。
实现这种方式时,首先需要定义一个2×2的矩阵,并且定义好矩阵的乘法以及乘方运算。当这些运算定义好了之后,剩下的事情就变得非常简单。完整的实现代码如下所示。
[cpp] 
<span style="font-family:Simsun;">#include<stdio.h> 
#include<assert.h> 
struct Matrix2By2 

    int m_00; 
    int m_01; 
    int m_10; 
    int m_11; 
    Matrix2By2(int m00=0,int m01=0,int m10=0,int m11=0):m_00(m00),m_01(m01),m_10(m10),m_11(m11) 
    { 
    } 
}; 
Matrix2By2 MatrixMultiply(const Matrix2By2& matrix1,const Matrix2By2& matrix2) 

    return Matrix2By2( 
            matrix1.m_00 * matrix2.m_00 + matrix1.m_01 * matrix2.m_10, 
            matrix1.m_00 * matrix2.m_01 + matrix1.m_01 * matrix2.m_11, 
            matrix1.m_10 * matrix2.m_00 + matrix1.m_11 * matrix2.m_10, 
            matrix1.m_10 * matrix2.m_01 + matrix1.m_11 * matrix2.m_11); 

Matrix2By2 MatrixPower(unsigned int n) 

    assert(n>0); 
    Matrix2By2 matrix; 
    if(n==1) 
        matrix=Matrix2By2(1,1,1,0); 
    else if(n%2==0) 
    { 
        matrix=MatrixPower(n/2); 
        matrix=MatrixMultiply(matrix,matrix); 
    } 
    else if(n%2==1) 
    { 
        matrix = MatrixPower((n - 1) / 2); 
        matrix = MatrixMultiply(matrix, matrix); 
        matrix = MatrixMultiply(matrix, Matrix2By2(1, 1, 1, 0)); 
    } 
    return matrix; 

int fibonacci_solution(unsigned int n) 

      int result[2] = {0, 1}; 
      if(n < 2) 
            return result[n]; 
 
      Matrix2By2 PowerNMinus2 = MatrixPower(n - 1); 
      return PowerNMinus2.m_00; 

void main() 

    unsigned int n; 
    printf("input n value:"); 
    scanf("%d",&n); 
    printf("%d\n",fibonacci_solution(n)); 
}</span> 
运行,能得到正确的结果。就像题目所说,给我fibonacci在我们计算机中的一个应用吧,就是跳台阶的问题。给定n个台阶,有两种选择,就是一次可以跳一个台阶,还有是一次可以跳两个台阶。求总共有多少总跳法,并分析算法的时间复杂度。把n级台阶时的跳法看成是n的函数,记为f(n)。当n>2时,第一次跳的时候就有两种不同的选择:一是第一次只跳1级,此时跳法数目等于后面剩下的n-1级台阶的跳法数目,即为f(n-1);另外一种选择是第一次跳2级,此时跳法数目等于后面剩下的n-2级台阶的跳法数目,即为f(n-2)。因此n级台阶时的不同跳法的总数f(n)=f(n-1)+(f-2)。
    我们把上面的分析用一个公式总结如下:
        /  1                             n=1
f(n)=      2  

补充:软件开发 , C++ ,
CopyRight © 2022 站长资源库 编程知识问答 zzzyk.com All Rights Reserved
部分文章来自网络,