分享

计算思维实践之路(四)

 宇哥小屋 2020-01-05
          什么是斐波那契数列?Fn+1=Fn+Fn-1,这个数列中的每个数字都是前两项数之和,如果是以1,1开头的自然数数列,1,1,2,3,5,8,13,21,34,55,89……这些数字被称为斐波那契数。同时,这个数列中还暗含着黄金比例,如果用数列中的每一个数字去除它后面的数字,数字越大,结果就越趋近于1.618,也就是我们平常所说的黄金比例。    
       科学家发现,一些植物的花瓣、萼片、果实的数目以及排列的方式上,都有一个神奇的规律,它们都非常符合著名的斐波那契数列。例如:蓟,它们的头部几乎呈球状。在右图中,你可以看到两条不同方向的螺旋。我们可以数一下,顺时针旋转的(和左边那条旋转方向相同)螺旋一共有13条,而逆时针旋转的则有21条。此外还有菊花、向日葵、松果、菠萝等都是按这种方式生长的。
     
      有些花瓣的数量和花序的排列确实体现出了斐波那契数列,但是大多数植物的花瓣和叶片排列并不会遵循这个原则。之所以出现斐波那契数和黄金比例的角度,都是能最有效利用空间的模型,而在不需要考虑空间使用的情况下,就会随机分布了,是否出现特别的数列,都与植物对生存环境的适应有密切关系。
      老猪用递归代码实现斐波那契:
           6 递归代码实现斐波那契
  1. #include<stdio.h>
  2. int fibonacci(int n) {
  3. if(n==1||n==2)//递归终止条件
  4. return 1;
  5. else//递归通式
  6. return (fibonacci(n-1)+fibonacci(n-2));
  7. }
  8. int main() {
  9. int n;
  10. printf("请输入1个整数:");
  11. scanf("%d",&n);
  12. printf("第%d个斐波那契是%d\n",n,fibonacci(n));
  13. return 0;
  14. }
            经常见到的其衍生题。比如一次登一个或两个台阶,问登n个台阶有多少种可能?青蛙一次跳一下或两下,问跳n下有几种可能?铺地砖问题等等都是使用斐波那契数列解决。

                 俺老孙试一试这个问题
       例7 一次登一个或两个台阶,问登n个台阶有多少种可能?
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. static int step[128];
  4. static int m = 1;
  5. int count(int n);
  6. void cout(int n,int t);
  7. int main()
  8. {
  9. int a,n;
  10. while(1) {
  11. printf("请输入需要登上的台阶数量:");
  12. scanf("%d",&n);
  13. a = count(n);
  14. printf("登上%d个台阶共有%d种方法!分别如下:\n",n,a);
  15. cout(n,0);
  16. m=1;
  17. }
  18. return 0;
  19. }
  20. int count(int n)
  21. {
  22. if(n == 1)
  23. return 1 ;
  24. if(n == 2)
  25. return 2 ;
  26. else
  27. return count(n - 1) + count(n - 2) ;
  28. }
  29. void cout(int n,int t)
  30. {
  31. int i,j;
  32. if(n<0)
  33. return ;
  34. if(n == 0)
  35. {
  36. printf("第%d种方法:",m);
  37. for(j=0; j<t; j++)
  38. printf("%d ",step[j]);
  39. m++;
  40. printf("\n");
  41. }
  42. else
  43. {
  44. for(i=1; i<=2; i++)
  45. {
  46. step[t] = i;
  47. cout(n-i,t+1);
  48. }
  49. }
  50. }
 
                      

         
                   例8 骨牌铺方格 在2×n的一个长方形方格中,用一个1× 2的骨牌铺满方格,输入n ,输出铺放方案的总数。例如n=3时,为2× 3方格,骨牌的铺放方案有三种,如下图:
      
                        
             Input
                输入数据由多行组成,每行包含一个整数n,表示该测试实例的长方形方格的规格是2×n (0<n<=50)。

            Output
          对于每个测试实例,请输出铺放方案的总数,每个实例的输出占一行。

          Sample Input
1 3 2


         Sample Output
1 3 2

            假设用arr[i]表示2*i的方格一共有组成的方法数,我们知道arr[1]=1;arr[2]=2;
       现在假设我们已经知道了arr[i-1]和arr[i-2],求arr[i],所谓arr[i],不过是在2*(i-1)的格子后边加上一格2*1的方格罢了,骨牌在这一格上横着放,竖着放,如果前面i-1块已经铺好,则第i块只有一种铺法,就是竖着放,如果要横着放,也只有一种铺法,不过要求前面i-2块已经铺好!
       因此arr[i]=arr[i-1]+arr[i-2];

  1. #include<iostream>
  2. using namespace std;
  3. int main() {
  4. __int64 arr[51];
  5. int num;
  6. arr[1]=1;
  7. arr[2]=2;
  8. for(int i=3; i<=50; i++)
  9. arr[i]=arr[i-1]+arr[i-2];
  10. while(scanf("%d",&num)!=EOF) {
  11. printf("%I64d\n",arr[num]);
  12. }
  13. return 0;
  14. }



    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多