为何你学不会递归?刷题几个月,告别递归,谈谈个人经验

可能不少人在大一的时候,就已经接触了递归了,不过,我敢保证不少人初学者刚开始接触递归的时候,是一脸懵逼的,我当初也是,给个人感受就是,递归太神奇了!数组

可能也有一大部分人知道递归,也能看的懂递归,但在实际作题过程当中,殊不知道怎么使用,有时候还容易被递归给搞晕。也有好几我的来问我有没有快速掌握递归的捷径啊。说实话,哪来那么多捷径啊,不过,我仍是想写一篇文章,谈谈个人一些经验,或许,可以给你带来一些帮助。bash

为了兼顾初学者,我会从最简单的题讲起!函数

递归的三大要素

第一要素:明确你这个函数想要干什么工具

对于递归,我以为很重要的一个事就是,这个函数的功能是什么,他要完成什么样的一件事,而这个,是彻底由你本身来定义的。也就是说,咱们先无论函数里面的代码什么,而是要先明白,你这个函数是要用来干什么。学习

例如,我定义了一个函数开发工具

// 算 n 的阶乘(假设n不为0)
int f(int n){
    
}
复制代码

这个函数的功能是算 n 的阶乘。好了,咱们已经定义了一个函数,而且定义了它的功能是什么,接下来咱们看第二要素。优化

第二要素:寻找递归结束条件ui

所谓递归,就是会在函数内部代码中,调用这个函数自己,因此,咱们必需要找出递归的结束条件,否则的话,会一直调用本身,进入无底洞。也就是说,咱们须要找出当参数为啥时,递归结束,以后直接把结果返回,请注意,这个时候咱们必须能根据这个参数的值,可以直接知道函数的结果是什么。spa

例如,上面那个例子,当 n = 1 时,那你应该可以直接知道 f(n) 是啥吧?此时,f(1) = 1。完善咱们函数内部的代码,把第二要素加进代码里面,以下3d

// 算 n 的阶乘(假设n不为0)
int f(int n){
    if(n == 1){
        return 1;
    }
}
复制代码

有人可能会说,当 n = 2 时,那咱们能够直接知道 f(n) 等于多少啊,那我能够把 n = 2 做为递归的结束条件吗?

固然能够,只要你以为参数是什么时,你可以直接知道函数的结果,那么你就能够把这个参数做为结束的条件,因此下面这段代码也是能够的。

// 算 n 的阶乘(假设n>=2)
int f(int n){
    if(n == 2){
        return 2;
    }
}
复制代码

注意我代码里面写的注释,假设 n >= 2,由于若是 n = 1时,会被漏掉,当 n <= 2时,f(n) = n,因此为了更加严谨,咱们能够写成这样:

// 算 n 的阶乘(假设n不为0)
int f(int n){
    if(n <= 2){
        return n;
    }
}
复制代码

第三要素:找出函数的等价关系式

第三要素就是,咱们要不断缩小参数的范围,缩小以后,咱们能够经过一些辅助的变量或者操做,使原函数的结果不变。

例如,f(n) 这个范围比较大,咱们可让 f(n) = n * f(n-1)。这样,范围就由 n 变成了 n-1 了,范围变小了,而且为了原函数f(n) 不变,咱们须要让 f(n-1) 乘以 n。

说白了,就是要找到原函数的一个等价关系式,f(n) 的等价关系式为 n * f(n-1),即

f(n) = n * f(n-1)。

这个等价关系式的寻找,能够说是最难的一步了,若是你不大懂也不要紧,由于你不是天才,你还须要多接触几道题,我会在接下来的文章中,找 10 道递归题,让你慢慢熟悉起来

找出了这个等价,继续完善咱们的代码,咱们把这个等价式写进函数里。以下:

// 算 n 的阶乘(假设n不为0)
int f(int n){
    if(n <= 2){
        return n;
    }
    // 把 f(n) 的等价操做写进去
    return f(n-1) * n;
}
复制代码

至此,递归三要素已经都写进代码里了,因此这个 f(n) 功能的内部代码咱们已经写好了。

这就是递归最重要的三要素,每次作递归的时候,你就强迫本身试着去寻找这三个要素。

仍是不懂?不要紧,我再按照这个模式讲一些题。

有些有点小基础的可能以为我写的太简单了,没耐心看?少侠,请继续看,我下面还会讲如何优化递归。固然,大佬请随意,能够直接拉动最下面留言给我一些建议,万分感谢!

案例1:斐波那契数列

斐波那契数列的是这样一个数列:一、一、二、三、五、八、1三、2一、34....,即第一项 f(1) = 1,第二项 f(2) = 1.....,第 n 项目为 f(n) = f(n-1) + f(n-2)。求第 n 项的值是多少。

一、第一递归函数功能

假设 f(n) 的功能是求第 n 项的值,代码以下:

int f(int n){
    
}
复制代码

二、找出递归结束的条件

显然,当 n = 1 或者 n = 2 ,咱们能够轻易着知道结果 f(1) =1, f(2) = 1。因此递归结束条件能够为 n <= 2。代码以下:

int f(int n){
    if(n <= 2){
        return 1;
    }
}
复制代码

第三要素:找出函数的等价关系式

题目已经把等价关系式给咱们了,因此咱们很容易就可以知道 f(n) = f(n-1) + f(n-2)。我说过,等价关系式是最难找的一个,而这个题目却把关系式给咱们了,这也太容易,好吧,我这是为了兼顾几乎零基础的读者。

因此最终代码以下:

int f(int n){
    // 1.先写递归结束条件
    if(n <= 2){
        return n;
    }
    // 2.接着写等价关系式
    return f(n-1) + f(n - 2);
}
复制代码

搞定,是否是很简单?

零基础的可能仍是不大懂,不要紧,以后慢慢按照这个模式练习!好吧,有大佬可能在吐槽太简单了。

案例2:小青蛙跳台阶

一只青蛙一次能够跳上1级台阶,也能够跳上2级。求该青蛙跳上一个n级的台阶总共有多少种跳法。

一、第一递归函数功能

假设 f(n) 的功能是求青蛙跳上一个n级的台阶总共有多少种跳法,代码以下:

int f(int n){
    
}
复制代码

二、找出递归结束的条件

我说了,求递归结束的条件,你直接把 n 压缩到很小很小就好了,由于 n 越小,咱们就越容易直观着算出 f(n) 的多少,因此当 n = 1时,你知道 f(1) 为多少吧?够直观吧?即 f(1) = 1。代码以下:

int f(int n){
    if(n == 1){
        return 1;
    }
}
复制代码

第三要素:找出函数的等价关系式

每次跳的时候,小青蛙能够跳一个台阶,也能够跳两个台阶,也就是说,每次跳的时候,小青蛙有两种跳法。

第一种跳法:第一次我跳了一个台阶,那么还剩下n-1个台阶还没跳,剩下的n-1个台阶的跳法有f(n-1)种。

第二种跳法:第一次跳了两个台阶,那么还剩下n-2个台阶还没,剩下的n-2个台阶的跳法有f(n-2)种。

因此,小青蛙的所有跳法就是这两种跳法之和了,即 f(n) = f(n-1) + f(n-2)。至此,等价关系式就求出来了。因而写出代码:

int f(int n){
    if(n == 1){
        return 1;
    }
    ruturn f(n-1) + f(n-2);
}
复制代码

你们以为上面的代码对不对?

答是不大对,当 n = 2 时,显然会有 f(2) = f(1) + f(0)。咱们知道,f(0) = 0,按道理是递归结束,不用继续往下调用的,但咱们上面的代码逻辑中,会继续调用 f(0) = f(-1) + f(-2)。这会致使无限调用,进入死循环

这也是我要和大家说的,关于递归结束条件是否够严谨问题,有不少人在使用递归的时候,因为结束条件不够严谨,致使出现死循环。也就是说,当咱们在第二步找出了一个递归结束条件的时候,能够把结束条件写进代码,而后进行第三步,可是请注意,当咱们第三步找出等价函数以后,还得再返回去第二步,根据第三步函数的调用关系,会不会出现一些漏掉的结束条件。就像上面,f(n-2)这个函数的调用,有可能出现 f(0) 的状况,致使死循环,因此咱们把它补上。代码以下:

int f(int n){
    //f(0) = 0,f(1) = 1,等价于 n<=1时,f(n) = n。
    if(n <= 1){
        return 1;
    }
    ruturn f(n-1) + f(n-2);
}
复制代码

有人可能会说,我不知道个人结束条件有没有漏掉怎么办?别怕,多练几道就知道怎么办了。

看到这里有人可能要吐槽了,这两道题也太容易了吧??能不能被这么敷衍。少侠,别走啊,下面出道难一点的。

下面其实也不难了,就比上面的题目难一点点而已,特别是第三步等价的寻找。

案例3:反转单链表。

反转单链表。例如链表为:1->2->3->4。反转后为 4->3->2->1

链表的节点定义以下:

class Node{
    int date;
    Node next;
}
复制代码

虽然是 Java语言,但就算你没学过 Java,我以为也是影响不大,能看懂。

仍是老套路,三要素一步一步来。

一、定义递归函数功能

假设函数 reverseList(head) 的功能是反转但链表,其中 head 表示链表的头节点。代码以下:

Node reverseList(Node head){
    
}
复制代码

2. 寻找结束条件

当链表只有一个节点,或者若是是空表的话,你应该知道结果吧?直接啥也不用干,直接把 head 返回呗。代码以下:

Node reverseList(Node head){
    if(head == null || head.next == null){
        return head;
    }
}
复制代码

3. 寻找等价关系

这个的等价关系不像 n 是个数值那样,比较容易寻找。可是我告诉你,它的等价条件中,必定是范围不断在缩小,对于链表来讲,就是链表的节点个数不断在变小,因此,若是你实在找不出,你就先对 reverseList(head.next) 递归走一遍,看看结果是咋样的。例如链表节点以下

咱们就缩小范围,先对 2->3->4递归下试试,即代码以下

Node reverseList(Node head){
    if(head == null || head.next == null){
        return head;
    }
    // 咱们先把递归的结果保存起来,先不返回,由于咱们还不清楚这样递归是对仍是错。,
    Node newList = reverseList(head.next);
}
复制代码

咱们在第一步的时候,就已经定义了 reverseLis t函数的功能能够把一个单链表反转,因此,咱们对 2->3->4反转以后的结果应该是这样:

咱们把 2->3->4 递归成 4->3->2。不过,1 这个节点咱们并无去碰它,因此 1 的 next 节点仍然是链接这 2。

接下来呢?该怎么办?

其实,接下来就简单了,咱们接下来只须要把节点 2 的 next 指向 1,而后把 1 的 next 指向 null,不就好了?,即经过改变 newList 链表以后的结果以下:

也就是说,reverseList(head) 等价于 ** reverseList(head.next)** + 改变一下1,2两个节点的指向。好了,等价关系找出来了,代码以下(有详细的解释):

//用递归的方法反转链表
public static Node reverseList2(Node head){
    // 1.递归结束条件
    if (head == null || head.next == null) {
             return head;
         }
         // 递归反转 子链表
         Node newList = reverseList2(head.next);
         // 改变 1,2节点的指向。
         // 经过 head.next获取节点2
         Node t1  = head.next;
         // 让 2 的 next 指向 2
         t1.next = head;
         // 1 的 next 指向 null.
        head.next = null;
        // 把调整以后的链表返回。
        return newList;
    }
复制代码

这道题的第三步看的很懵?正常,由于你作的太少了,可能没有想到还能够这样,多练几道就能够了。可是,我但愿经过这三道题,给了你之后用递归作题时的一些思路,你之后作题能够按照我这个模式去想。经过一篇文章是不可能掌握递归的,还得多练,我相信,只要你认真看个人这篇文章,多看几回,必定能找到一些思路!!

我已经强调了好屡次,多练几道了,因此呢,后面我也会找大概 10 道递归的练习题供你们学习,不过,我找的可能会有必定的难度。不会像今天这样,比较简单,因此呢,初学者还得本身多去找题练练,相信我,掌握了递归,你的思惟抽象能力会更强!

接下来我讲讲有关递归的一些优化。

有关递归的一些优化思路

1. 考虑是否重复计算

告诉你吧,若是你使用递归的时候不进行优化,是有很是很是很是多的子问题被重复计算的。

啥是子问题? f(n-1),f(n-2)....就是 f(n) 的子问题了。

例如对于案例2那道题,f(n) = f(n-1) + f(n-2)。递归调用的状态图以下:

看到没有,递归计算的时候,重复计算了两次 f(5),五次 f(4)。。。。这是很是恐怖的,n 越大,重复计算的就越多,因此咱们必须进行优化。

如何优化?通常咱们能够把咱们计算的结果保证起来,例如把 f(4) 的计算结果保证起来,当再次要计算 f(4) 的时候,咱们先判断一下,以前是否计算过,若是计算过,直接把 f(4) 的结果取出来就能够了,没有计算过的话,再递归计算。

用什么保存呢?能够用数组或者 HashMap 保存,咱们用数组来保存把,把 n 做为咱们的数组下标,f(n) 做为值,例如 arr[n] = f(n)。f(n) 尚未计算过的时候,咱们让 arr[n] 等于一个特殊值,例如 arr[n] = -1。

当咱们要判断的时候,若是 arr[n] = -1,则证实 f(n) 没有计算过,不然, f(n) 就已经计算过了,且 f(n) = arr[n]。直接把值取出来就好了。代码以下:

// 咱们实现假定 arr 数组已经初始化好的了。
int f(int n){
    if(n <= 1){
        return n;
    }
    //先判断有没计算过
    if(arr[n] != -1){
        //计算过,直接返回
        return arr[n];
    }else{
        // 没有计算过,递归计算,而且把结果保存到 arr数组里
        arr[n] = f(n-1) + f(n-1);
        reutrn arr[n];
    }
}
复制代码

也就是说,使用递归的时候,必要 需要考虑有没有重复计算,若是重复计算了,必定要把计算过的状态保存起来。

2. 考虑是否能够自底向上

对于递归的问题,咱们通常都是从上往下递归的,直到递归到最底,再一层一层着把值返回。

不过,有时候当 n 比较大的时候,例如当 n = 10000 时,那么必需要往下递归10000层直到 n <=1 才将结果慢慢返回,若是n太大的话,可能栈空间会不够用。

对于这种状况,其实咱们是能够考虑自底向上的作法的。例如我知道

f(1) = 1;

f(2) = 2;

那么咱们就能够推出 f(3) = f(2) + f(1) = 3。从而能够推出f(4),f(5)等直到f(n)。所以,咱们能够考虑使用自底向上的方法来取代递归,代码以下:

public int f(int n) {
       if(n <= 2)
           return n;
       int f1 = 1;
       int f2 = 2;
       int sum = 0;

       for (int i = 3; i <= n; i++) {
           sum = f1 + f2;
           f1 = f2;
           f2 = sum;
       }
       return sum;
   }
复制代码

这种方法,其实也被称之为递推

最后总结

其实,递归不必定老是从上往下,也是有不少是从下往上的,例如 n = 1 开始,一直递归到 n = 1000,例如一些排序组合。对于这种从下往上的,也是有对应的优化技巧,不过,我就先不写了,后面再慢慢写。这篇文章写了好久了,脖子有点受不了了,,,,颈椎病?惧怕。。。。

说实话,对于递归这种比较抽象的思想,要把他讲明白,特别是讲给初学者听,仍是挺难的,这也是我这篇文章用了很长时间的缘由,不过,只要能让大家看完,有所收获,我以为值得!有些人可能以为讲的有点简单,没事,我后面会找一些不怎么简单的题。最后若是以为不错,还请给我转发 or 点赞一波!

若是你以为这篇内容对你挺有启发,我想邀请你帮我三个忙,让更多的人看到这篇文章:

一、点赞,让更多的人也能看到这篇内容(收藏不点赞,都是耍流氓 -_-)

二、关注我和专栏,让咱们成为长期关系

三、关注公众号「苦逼的码农」,里面已有100多篇原创文章,我也分享了不少视频、书籍的资源,以及开发工具,欢迎各位的关注,第一时间阅读个人文章。

相关文章
相关标签/搜索