怎么更好地终极理解递归算法

 递归真是个奇妙的思维方式。对一些简单的递归问题,我总是惊叹于递归描述问题和编写代码的简洁。但是总感觉没能融会贯通地理解递归,有时尝试用大脑去深入“递归”,层次较深时便常产生进不去,出不来的感觉。这种状态也导致我很难灵活地运用递归解决问题。有一天,我看到一句英文:“To Iterate is Human, to Recurse, Divine.”中文译为:“人理解迭代,神理解递归。”然后,我心安理得地放弃了对递归的深入理解。直到看到王垠谈程序语言最精华的原理时提到了递归,并说递归比循环表达能力强很多,而且效率几乎一样。再次唤醒了我对递归的理解探索。


    我首先在知乎上发现了下面两个例子,对比了递归和循环。

    递归:你打开面前这扇门,看到屋里面还有一扇门(这门可能跟前面打开的门一样大小(静),也可能门小了些(动)),你走过去,发现手中的钥匙还可以打开它,你推开门,发现里面还有一扇门,你继续打开,。。。, 若干次之后,你打开面前一扇门,发现只有一间屋子,没有门了。 你开始原路返回,每走回一间屋子,你数一次,走到入口的时候,你可以回答出你到底用这钥匙开了几扇门。

    循环:你打开面前这扇门,看到屋里面还有一扇门,(这门可能跟前面打开的门一样大小(静),也可能门小了些(动)),你走过去,发现手中的钥匙还可以打开它,你推开门,发现里面还有一扇门,(前面门如果一样,这门也是一样,第二扇门如果相比第一扇门变小了,这扇门也比第二扇门变小了(动静如一,要么没有变化,要么同样的变化)),你继续打开这扇门,。。。,一直这样走下去。 入口处的人始终等不到你回去告诉他答案。


该用户这么总结到:

    递归就是有去(递去)有回(归来)。


具体来说,为什么可以”有去“? 

    这要求递归的问题需要是可以用同样的解题思路来回答除了规模大小不同其他完全一样的问题。


为什么可以”有回“?

  这要求这些问题不断从大到小,从近及远的过程中,会有一个终点,一个临界点,一个baseline,一个你到了那个点就不用再往更小,更远的地方走下去的点,然后从那个点开始,原路返回到原点。


上面的解释几乎回答了我已久的疑问:为什么我老是有递归没有真的在解决问题的感觉?

    因为递是描述问题,归是解决问题。而我的大脑容易被递占据,只往远方去了,连尽头都没走到,何谈回的来。


《漫谈递归:递归的思想》这篇文章将递归思想归纳为:

    递归的基本思想是把规模大的问题转化为规模小的相似的子问题来解决。在函数实现时,因为解决大问题的方法和解决小问题的方法往往是同一个方法,所以就产生了函数调用它自身的情况。另外这个解决问题的函数必须有明显的结束条件,这样就不会产生无限递归的情况了。


    需注意的是,规模大转化为规模小是核心思想,但递归并非是只做这步转化,而是把规模大的问题分解为规模小的子问题和可以在子问题解决的基础上剩余的可以自行解决的部分。而后者就是归的精髓所在,是在实际解决问题的过程。


    我试图把我理解到递归思想用递归用程序表达出来,确定了三个要素:递 + 结束条件 + 归。

function recursion(大规模)
{
    if (end_condition)
    {
        end;     
    }
    else
    {     //先将问题全部描述展开,再由尽头“返回”依次解决每步中剩余部分的问题
        recursion(小规模);     //go;
        solve;                //back;
    }
}

    但是,我很容易发现这样描述遗漏了我经常会遇到的一种递归情况,比如递归遍历的二叉树的先序。我将这种情况用如下递归程序表达出来。

function recursion(大规模)
{
    if (end_condition)
    {
        end;     
    }
    else
    {     //在将问题转换为子问题描述的每一步,都解决该步中剩余部分的问题。
        solve;                //back;
        recursion(小规模);     //go;
    }
}

    总结到这里,我突然发现递归是为了最能表达这种思想,所以用“递归”这个词,其实递归可以是“有去有回”,也可以是“有去无回”。但其根本是“由大往小地去,由近及远地去”。“递”是必需,“归”并非必需,依赖于要解决的问题,有的需要去的路上解决,有的需要回来的路上解决。有递无归的递归其实就是我们很容易理解的一种分治思想。


    其实理解递归可能没有“归”,只有去(分治)的情况后,我们应该想到递归也许可以既不需要在“去”的路上解决问题,也不需要在“归”的路上解决问题,只需在路的尽头解决问题,即在满足停止条件时解决问题。递归的分治思想不一定是要把问题规模递归到最小,还可以是将问题递归穷举其所有的情形,这时通常递归的表达力体现在将无法书写的嵌套循环(不确定数量的嵌套循环)通过递归表达出来。

将这种递归情形用递归程序描述如下:

recursion()
{
    if (end_condition)
    {
        solve;     
    }
    else
    {     //在将问题转换为子问题描述的每一步,都解决该步中剩余部分的问题。
        for () { recursion();     //go; }
    }
}


    由这个例子,可以发现这种递归对递归函数参数出现了设计要求,即便递归到尽头,组合的字符串规模(长度)也没有变小,规模变小的是递归函数的一个参数。可见,这种变化似乎一下将递归的灵活性大大地扩展了,所谓的大规模转换为小规模需要有一个更为广义的理解了。


    对递归的理解就暂时到这里了,可以看出文章中提到关于“打开一扇门”的递归例子来解释递归并不准确,例子只描述了递归的一种情况。而“递归就是有去(递去)有回(归来)”的论断同样不够准确。要为只读了文章前半部分的读者惋惜了。我也给出自己对递归思想的总结吧:


    递归的基本思想是广义地把规模大的问题转化为规模小的相似的子问题或者相似的子问题集合来解决。广义针对规模的,规模的缩小具体可以是指递归函数的参数,也可以是其参数之一。相似是指解决大问题的方法和解决小问题的方法往往是同一个方法,还可以是指解决子问题集的各子问题的方法是同一个方法。解决大问题的方法可以是由解决次规模问题的方法和解决剩余部分的方法组成,也可以是由一系列解决次规模问题的方法组成。

  • 41
    点赞
  • 80
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
分治和递归算法是计算机科学中的两种重要算法思想,它们可以帮助我们解决很多复杂的问题。下面我介绍一下这两种算法以及它们在Java中的实现。 1. 分治算法 分治算法是指将一个大问题分成若干个小问题,然后逐个解决这些小问题,最后将所有小问题的解合并起来得到大问题的解。分治算法通常采用递归的方式来实现。 Java代码示例: ```java public int divideAndConquer(int[] nums, int left, int right){ if(left == right) return nums[left]; // 如果数组只有一个元素,则直接返回该元素 int mid = (left + right) / 2; // 将数组分成两部分 int leftSum = divideAndConquer(nums, left, mid); // 递归求解左半部分的和 int rightSum = divideAndConquer(nums, mid + 1, right); // 递归求解右半部分的和 return leftSum + rightSum; // 合并左右两部分的和 } ``` 上面的代码实现了求解一个数组中所有元素的和的问题。首先将数组分成两部分,然后递归地求解左右两部分的和,最后将左右两部分的和合并起来得到整个数组的和。 2. 递归算法 递归算法是指在函数的定义中调用自身的算法。递归算法通常用于解决具有递归结构的问题,比如树和图等数据结构。递归算法需要满足两个条件:基本情况和递归情况。 Java代码示例: ```java public int fibonacci(int n){ if(n <= 1) return n; // 基本情况 return fibonacci(n-1) + fibonacci(n-2); // 递归情况 } ``` 上面的代码实现了求解斐波那契数列的第n项的值的问题。斐波那契数列中的第一项和第二项都是1,从第三项开始,每一项都是前两项的和。递归算法中,基本情况是当n小于等于1时直接返回n,递归情况是求解前两项的和,然后递归求解前两项中的每一项。 希望这些示例代码能够帮助你理解分治和递归算法的基本思想和Java实现方式。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值