zoukankan      html  css  js  c++  java
  • 我对递归的理解(适合新手)

    一个函数调用其它函数好理解,但一个函数调用自身函数(这就是递归),代码看起来就不好懂了,容易把人绕进去出不来。不就是一个函数吗,为什么会让人看不懂呢?因为理解递归应该从整体(即整个函数体是否能解决问题)来看,而不是纠结于递归函数执行到了哪里!这是我研究了一遍又一遍最简单的递归代码得出得结论。我曾把二叉树的前序遍历(递归形式)的每一步代码都写出来,用python写的,因为python每次调用函数就缩进,能直观看出执行到哪了,结果发现递归确实能遍历,而且是按照可以从二叉树图形的每个结点顺序,其实也是,代码的执行不就是遍历了每个结点吗?要是代码没有按照设想的逻辑运行,那这个函数不就没实现功能,就是错的代码吗?
    递归问题具体表现就是在函数中调用自身函数,每次调用问题的规模都变小了,直到遇到base case,一层层返回(可能是直接调用return,也可能函数体执行完返回的)问题就解决了,因此递归问题必须可以分解为若干个规模较小、与原问题形式相同的子问题,这些子问题可以用相同的解题思路来解决,这样才能从开始传一个参数,然后执行过程中不断调用自身函数直到最后得到问题的解。
    拿斐波那契问题的代码来分析:

    long long Fib(int n)
    {
          if(n <= 0)
                return 0;
          if(n == 1)
                return 1;
          return Fib(n-1)+Fib(n-2);
    }
    


    由斐波那契数列的递归代码可知,每个数都是f(n-1)+f(n-2),其base case是当n=0是f(n)=0,当n=1时,f(n)=1。函数也确实就是这样实现的,那为什么新手阅读代码还会理解不了?当看到return Fib(n-1)+Fib(n-2)时,新手会去想Fib(n-1)和Fib(n-2)的计算过程,多重复这个过程几遍就把自己绕进去了,不知道计算到Fib()哪个。如下图所示,求n=10的斐波那契数列,n=10作为参数传进函数,递归后就需计算n=9和n=8;而n=9递归后计算n=8和n=7,n=8递归后计算n=7和n=6,n=9是n=8、7、6、5、4、3、2、1计算来的,n=8是n=7、6、5、4、3、2、1计算来的,是分别单独递归计算得到,用递归的树型结构可直观看出,数的每个结点都计算了,怎么计算来的,由更底下的结点推算而来到最底下的结点即叶子,不是f(0)就是f(1)。

    图 基于递归求斐波那契数列的第10项的调用过程
    图 基于递归求斐波那契数列的第10项的调用过程
    正确的理解是:f(n)=f(n-1)+f(n-2),f(n-1)=f(n-2)+f(n-3),f(n-2)=f(n-3)+f(n-4)…每次调用Fib()函数,传入的参数都能正确计算,说明函数体没问题。这就是从整体来看的思维,每次调用Fib()函数都正确,递归直到base case才返回(return),问题就解决了。
    根据斐波那契数列,我们可以提取出递归的3要素:
    1)、明确递归终止条件;
    2)、给出递归终止时的处理办法;
    3)、提取重复的逻辑,缩小问题规模。
    再举一个例子:青蛙跳台阶问题。一只青蛙一次可以跳上1级台阶,也可以跳上2级台阶。求该青蛙跳上一个n级的台阶总共有多少种跳法。
    只有1级台阶时,只有1种跳法,即跳上1级台阶;
    有2级台阶时,有2种跳法,连续2次跳1级台阶,或者直接跳上2级台阶;
    有3级台阶时,有3种跳法:第一种是直接连续跳3次1级台阶,第二种是先跳1级再跳2级,第三种是先跳2级,再跳1级;
    有4级台阶时,有5种跳法;
    有5级台阶时,有8种跳法;

    3=2+1,5=3+2,8=5+3…
    直觉好的人已经归纳出数学计算公式:f(n)=f(n-1)+f(n-2)。
    另一个思考角度:把n级台阶的跳法看成n的函数,记为f(n)。我们从后往前看,首先考虑n级台阶,只能由第n-1级台阶跳1级上来,或是由第n-2级台阶跳2级; n-1级台阶时,只能由第n-2级跳1级上来,或由n-3级跳2级;
    n-2级台阶时,只能由第n-3级跳1级上来,或由n-4级跳2级;

    2级台阶时,只能由第1级跳1级上来,或由0级跳2级;
    1级台阶时,只能由第0级跳1级上来;
    就可以归纳得出:

    [ f(n)=egin{cases} 1 & n=1 \ 2 & n=2 \ f(n-1)+f(n-2) & n>2 end{cases}]

    这不就是斐波那契数列问题吗?
    至于为什么可以把n级台阶的跳法看成f(n),第n级的跳法是由n-1级的跳法加上n-2级的跳法,一次类推。因为一种从0级跳到终点的路径就是一种跳法,如果画出树形结构,就能看到一条条从叶结点到根的路径,每一条路径都是一种跳法,即有多少叶结点就有多少种跳法,这就是为什么跳法可以用f(n)表示的原因。

  • 相关阅读:
    centos shell脚本编程1 正则 shell脚本结构 read命令 date命令的用法 shell中的逻辑判断 if 判断文件、目录属性 shell数组简单用法 $( ) 和${ } 和$(( )) 与 sh -n sh -x sh -v 第三十五节课
    基于HTML5 WebGL实现 json工控风机叶轮旋转
    基于HTML5的WebGL实现的2D3D迷宫小游戏
    基于HTML5和WebGL的碰撞测试
    基于HTML5和WebGL的3D网络拓扑结构图
    基于 HTML5 WebGL 的 3D 网络拓扑图
    基于HTML5 Canvas 实现弹出框
    基于HTML5 Canvas实现用户交互
    基于HTML5快速搭建TP-LINK电信拓扑设备面板
    HTML5 技术在风电、光伏等新能源领域的应用
  • 原文地址:https://www.cnblogs.com/jiangzhongzhiwei/p/13255948.html
Copyright © 2011-2022 走看看