• Java 算法篇-深入理解递归(递归实现:青蛙爬楼梯)


    🔥博客主页: 小扳_-CSDN博客
    ❤感谢大家点赞👍收藏⭐评论✍
     

     

    文章目录

            1.0 递归的说明

            2.0 用递归来实现相关问题

            2.1 递归 - 阶乘

            2.2 递归 - 反向打印字符串

            2.3 递归 - 二分查找

            2.4 递归 - 冒泡排序

            2.5 递归 - 冒泡排序2.0

            2.6 递归 - 插入排序

            2.7 递归 - 斐波那契

            2.8 递归 - 兔子问题

            2.9 递归 - 青蛙爬楼梯


            1.0 递归的说明

            递归就是在一个函数中调用自身。这样做可以让我们解决一些问题,比如计算斐波那契数列、阶乘等。

            递归函数一般包括两部分:基本情况和递归情况。基本情况是指当问题变得很小,可以直接得到答案时,递归就可以停止了。递归情况是指在解决问题的过程中,需要不断地调用自身来解决更小规模的问题。

            对于递归这个算法,简单的来说,方法自身调用自身的时候,需要有终止的条件,在运行过程中不断的趋向终止条件。还有递归总的来说有两个动作:第一个动作是递出,方法不断的在栈区中创建出来,直到达到了条件就会停止。第二个动作,达到条件停止了,就会回归,指方法在栈区中依次执行完后就销毁。

            2.0 用递归来实现相关问题

            以下的问题都较为简单,采取直接用代码来演示。

            2.1 递归 - 阶乘

    代码如下:

    1. //阶乘
    2. public static void main(String[] args) {
    3. System.out.println(fun(5));
    4. }
    5. public static int fun(int n) {
    6. if (n == 1) {
    7. return 1;
    8. }
    9. return n * fun(n-1);
    10. }
    11. }

    运行结果为:

            2.2 递归 - 反向打印字符串

    代码如下:

    1. //反向打印字符串
    2. public static void main(String[] args) {
    3. String str = "lisi";
    4. fun2(str,0);
    5. }
    6. public static void fun2 (String s, int n) {
    7. if (n == s.length()) {
    8. return;
    9. }
    10. fun2(s,n + 1);
    11. System.out.println(s.charAt(n));
    12. }

    运行结果:

            2.3 递归 - 二分查找

    代码如下:

    1. //二分查找
    2. public static void main(String[] args) {
    3. int[] arr = {1,3,5,7,9,10,13};
    4. System.out.println(fun3(arr, 0, arr.length - 1, 4));
    5. }
    6. public static int fun3 (int[] arr, int left, int right, int target) {
    7. int mid = (left + right) >>> 1;
    8. if (left > right) {
    9. return -1;
    10. }
    11. if(arr[mid] < target) {
    12. return fun3(arr, mid + 1,right,target);
    13. } else if (target < arr[mid]) {
    14. return fun3(arr,left,right - 1,target);
    15. }else {
    16. return mid;
    17. }
    18. }

      运行结果如下:

            没有找到就返回 - 1

            2.4 递归 - 冒泡排序

    代码如下:

    1. //冒泡排序
    2. public static void main(String[] args) {
    3. int[] arr = {1,5,2,4,9,1,3};
    4. fun4(arr, arr.length - 1);
    5. System.out.println(Arrays.toString(arr));
    6. }
    7. public static void fun4 (int[] arr, int n) {
    8. if (n == 0) {
    9. return;
    10. }
    11. for (int i = 0; i < n; i++) {
    12. if (arr[i] > arr[i + 1]) {
    13. int temp = arr[i];
    14. arr[i] = arr[i+1];
    15. arr[i+1] = temp;
    16. }
    17. }
    18. fun4(arr,n-1);
    19. }

    运行结果如下:

            2.5 递归 - 冒泡排序2.0

            对冒泡排序进行升级,假如 int[] arr = {2,1,1,3,4,5,9},这种只需要遍历一遍即可,但是对与已经用递归实现的冒泡不止遍历一次。因此,需要得到升级版冒泡排序。

            思路为:对于后续的元素已经是排好序了,就不用再遍历了。每一次交换完元素之后记下来 i 索引,i 之后的元素已经是排好序的,i 之前的元素还需要继续遍历,看是否还需要交换。

    代码如下:

    1. //冒泡排序升级版
    2. public static void main(String[] args) {
    3. int[] arr = {1,3,2,4,9,10,13};
    4. fun4(arr, arr.length - 1);
    5. System.out.println(Arrays.toString(arr));
    6. }
    7. public static void fun4 (int[] arr, int n) {
    8. if (n == 0) {
    9. return;
    10. }
    11. int j = 0;
    12. for (int i = 0; i < n; i++) {
    13. if (arr[i] > arr[i + 1]) {
    14. int temp = arr[i];
    15. arr[i] = arr[i+1];
    16. arr[i+1] = temp;
    17. j = i;
    18. }
    19. }
    20. fun4(arr,j);
    21. }

            如果还不是很清晰的话,可以一步步来调试一下,来对比两种冒泡的执行过程。

            2.6 递归 - 插入排序

            思路:假设第一个元素已经排序好了的,在已经排好的元素的后一个元素记录为 low,这个 low 索引对应的元素需要用临时变量来接受,只要找到比这个索引对应的元素小的值,就可以插入到比它小的值的后一个索引位置了,当然,每一次对比之后,都需要往后移一个位置,以便直接插入。当 low 一直每一个加 1 ,当 low 等于数组的长度时,就该停止了继续递归下去了。

    代码如下:

    1. public class Recursion {
    2. // 插入排序
    3. public static void main(String[] args) {
    4. int[] arr = {1,3,2,4,9,10,13};
    5. fun5(arr,1);
    6. System.out.println(Arrays.toString(arr));
    7. }
    8. public static void fun5 (int[] arr,int low) {
    9. if (low == arr.length) {
    10. return;
    11. }
    12. int temp = arr[low];
    13. int i = low - 1;
    14. while (arr[i] > temp) {
    15. arr[i + 1] = arr[i];
    16. i--;
    17. }
    18. arr[i + 1] = temp;
    19. fun5(arr,low + 1);
    20. }

    运行结果如下:

            2.7 递归 - 斐波那契

    代码如下:

    1. //斐波那契
    2. public static void main(String[] args) {
    3. System.out.print(fun6(1) +" ");
    4. System.out.print(fun6(2) +" ");
    5. System.out.print(fun6(3) +" ");
    6. System.out.print(fun6(4) +" ");
    7. System.out.print(fun6(5) +" ");
    8. System.out.print(fun6(6) +" ");
    9. }
    10. public static int fun6 (int n) {
    11. if (n == 0) {
    12. return 0;
    13. }
    14. if (n == 1 || n == 2) {
    15. return 1;
    16. }
    17. return fun6(n-1) + fun6(n - 2);
    18. }

    运行结果如下:

            2.8 递归 - 兔子问题

            一个斐波那契的变体问题。

            思路:观察第六个月的兔子个数,是否等于第四个月的兔子的总数加上第五个月的兔子总数;类推,第五个月的兔子个数,是否等于第四个月的兔子的总数加上第三个月的兔子总数;以此类推,是符合斐波那契逻辑的。

    代码如下:

    1. //兔子问题
    2. public static void main(String[] args) {
    3. System.out.print(fun7(1) + " ");
    4. System.out.print(fun7(2) + " ");
    5. System.out.print(fun7(3) + " ");
    6. System.out.print(fun7(4) + " ");
    7. System.out.print(fun7(5) + " ");
    8. }
    9. public static int fun7 (int n) {
    10. if (n == 1) {
    11. return 1;
    12. }
    13. if (n == 0) {
    14. return 0;
    15. }
    16. return fun7(n -1) + fun7(n - 2);
    17. }

    运行结果如下:

            2.9 递归 - 青蛙爬楼梯

            一个斐波那契的变体问题。

    题目如下:

    列举一下:

            实现思路: 一个阶梯一种跳法,两个阶梯两种跳法。重点,如果有四个阶梯,从后往前分析,分两种情况;第一种,从第二个台阶直接一下子跳两阶上来。第二种,从第三个台阶跳一阶上来。那么从考虑第一种情况,前面两阶是不是就是只有两种方法。考虑第二种情况,前面的三个台阶是不是就是前面已经算出来的方式跳法个数了。因此,这就是一个斐波那契的变体问题。

    代码如下:

    1. //青蛙问题
    2. public static void main(String[] args) {
    3. System.out.print(fun8(1) + " ");
    4. System.out.print(fun8(2) + " ");
    5. System.out.print(fun8(3) + " ");
    6. System.out.print(fun8(4) + " ");
    7. System.out.print(fun8(5) + " ");
    8. System.out.print(fun8(6) + " ");
    9. }
    10. public static int fun8 (int n) {
    11. if (n == 1) {
    12. return 1;
    13. }
    14. if (n == 2) {
    15. return 2;
    16. }
    17. return fun8(n-1) +fun8(n-2);
    18. }

    运行结果如下:

  • 相关阅读:
    torch 的数据加载 Datasets & DataLoaders
    java计算机毕业设计小王防疫副食品配送商城源程序+mysql+系统+lw文档+远程调试
    微机原理_9
    动态网页和前端技术基础知识
    淘宝API官方商品、交易、订单、物流、插旗接口如下:
    侧边栏的文章分类、热门文章和热门文章的展示(Go 搭建 qiucode.cn 之九)
    计算机毕业设计选题推荐-springboot 小说阅读平台
    软件设计模式系列之二十四——模板方法模式
    pip换源加速下载
    window删除文件夹时提示源路径太长无法删除的解决办法
  • 原文地址:https://blog.csdn.net/Tingfeng__/article/details/134325079