• 【JavaSE】/*基于Java语言的递归运算*/


    ✨hello,进来的小伙伴们,你们好呐!✨

    🍊系列专栏:【JavaSE】🍱

    🍎本篇内容:JavaSE部分的递归运算详解!(对准知识点阅读喔!)

    🍮作者简介:一名大二即将大三的编程小白,我很平凡,学会努力!

    🍻码云存放仓库gitee:https://gitee.com/king-zhou-of-java/java-se.git

    ✈️✈️正文开始:

    🍏递归定义:一个方法在执行过程中调用自身, 就称为 "递归"。

    🍎递归的必要条件:
    1. 将原问题划分成其子问题,注意:子问题必须要与原问题的解法相同。
    2. 递归出口。

    🍎🍎目录

    一、实例演示:递归求N的阶乘

    递归过程分析:

    二、 递归调用练习

    一、递归求1+2+3+……10的和

    二、顺序打印一个数字的每一位

    三、返回一个数组成本身的数字之和

    四、求解汉诺塔问题

    五、求斐波那契数列第N项


    一、实例演示:递归求N的阶乘

    1. public class fac {
    2. public static int factorial(int x){
    3. if(x<2){
    4. return 1;
    5. }
    6. else{
    7. return x * factorial(x-1);//递归调用本身
    8. }
    9. }
    10. public static void main(String[] args) {
    11. Scanner sc = new Scanner(System.in);
    12. int n = sc.nextInt();
    13. System.out.println(factorial(n));
    14. }
    15. }

    🍔递归过程分析:

    🍖🍖本题假设我们想求解5的阶乘,我们可以看到我们从main函数里面输入一个数N,这里我们输入5,随即在我们的功能函数factorial接收到参数5,接着因为if里面的条件是x<2,不满足,所以执行我们的else里面的语句,我们发现是return x * factorial(x-1);我们输入的是5,所以即 return 5   *factorial(4);同理我们调用了本身这个factorial函数,传进去的参数是4,接着继续……,直到我们的参数变成1<2,那么这时递归的  “递” ,结束,开始我们的 “归”。🛥️🛥️

    1. // 执行结果
    2. 函数开始, n = 5
    3. 函数开始, n = 4
    4. 函数开始, n = 3
    5. 函数开始, n = 2
    6. 函数开始, n = 1
    7. 函数结束, n = 1 ret = 1
    8. 函数结束, n = 2 ret = 2
    9. 函数结束, n = 3 ret = 6
    10. 函数结束, n = 4 ret = 24
    11. 函数结束, n = 5 ret = 120
    12. ret = 120

    🍶🍶运行结果:

    aa84e91ee535421abf6831be889e80ad.png

    二、 递归调用练习

    🍓🍓一、递归求1+2+3+……10的和

    1. public class result {
    2. public static int fun(int n){
    3. if(n==1){
    4. return 1;
    5. }
    6. return n+fun(n-1);
    7. }
    8. public static void main(String[] args) {
    9. Scanner scanner = new Scanner(System.in);
    10. int n = scanner.nextInt();
    11. System.out.println(fun(n));
    12. }
    13. }

    🍇🍇递归的核心思想就是我们的递归体应该如何设计,本题我们想得到1+……10的和,来看我们的递归体如何设计的!

    🍨运行结果:

    95f3e5dca8c24d179cc37374acd891e7.png

     🍊二、顺序打印一个数字的每一位

    🍨问题分析:比如我们想打印1234的每一位,那么打印出来应该就是1 2 3 4那么首先就是如何判断我们输入的数字是几位数,看下面的功能代码部分,设计非常的巧妙,通过是否n>9,是->我们递归调用本身传参数 “n/10”,打印的结果就是  n%10  这样肯定得到的就是我们的每一位数字!⛵⛵

    1. public class print {
    2. public static void fun(int n){
    3. if(n>9){
    4. fun(n/10);
    5. }
    6. System.out.print(n%10+" ");
    7. }
    8. public static void main(String[] args) {
    9. Scanner sc = new Scanner(System.in);
    10. int n = sc.nextInt();
    11. fun(n);
    12. }
    13. }

    🍬🍬运行结果:

    63bd26a4efac46a9b3d2e5c3c5054336.png

    🍮 三、返回一个数组成本身的数字之和

    比如我们输入1234,输出就是1+2+3+4=10。

    🍖🍖函数实现:

    1. public class sum {
    2. public static int sumd(int num) {
    3. if (num < 10)
    4. return num;
    5. return num % 10 + sumd(num / 10);
    6. }
    7. public static void main(String[] args) {
    8. Scanner sc= new Scanner(System.in);
    9. int n = sc.nextInt();
    10. System.out.println(sumd(n));
    11. }
    12. }

    🍓🍓运行结果:

    5336858523764227921bf9d8ea1f859c.png

    🛩️🛩️四、求解汉诺塔问题

    ✨定义:汉诺塔(Tower of Hanoi),又称河内塔,是一个源于印度古老传说的益智玩具大梵天创造世界的时候做了三根金刚石柱子,在一根柱子上从下往上按照大小顺序摞着64片黄金圆盘。大梵天命令婆罗门把圆盘从下面开始按大小顺序重新摆放在另一根柱子上。并且规定,在小圆盘上不能放大圆盘,在三根柱子之间一次只能移动一个圆盘。🌯🌯

    ⛵⛵代码实现:

    1. public class Hanio {
    2. public static void han(int n,char pos1,char pos2,char pos3){
    3. if(n==1){
    4. move(pos1,pos3);
    5. return;
    6. }
    7. han(n-1,pos1,pos3,pos2);
    8. move(pos1,pos3);
    9. han(n-1,pos2,pos1,pos3);
    10. }
    11. public static void move(char pos1,char pos2){
    12. System.out.println(pos1+"->"+pos2);
    13. }
    14. public static void main(String[] args) {
    15. han(3,'A','B','C');
    16. }
    17. }

    ⛴️⛴️代码解读:通过定义我们可以了解到每次只能移动一个盘子,并且小盘子要放在大盘子上面,那么这里我们有A B C,三个圆柱,我们可以将其依次理解为:初始位置   跳板位置  目标位置,我们看函数部分,如果只有一个盘子我们直接从A->C 只需移动一步便可,那么>1的情况,这里我们假设要移动三个盘子,通过画图我们可以发现首先要将2个盘子移动到B圆柱再借助A移动到C盘,那么这里的第一次调用 han(n-1,pos1,pos3,pos2);我们便可以理解,下次递归将(n-1)作为盘子个数,pos1就是我们的起始位置,pos3就是我们的跳板位置,pos2就是我们的目标位置,因为首先我们将(n-1)个盘子放在了B(pos2)上,调用结束后,执行我们的move函数,输出我们这次的移动轨迹,下次调用就是han(n-1,pos2,pos1,pos3);同理,这个时候pos2就是我们的起始位置,pos1变成我们的跳板位置,最后pos3是我们的目标位置。🍱🍱

    🍇🍇运行结果:(我们可以自己画图尝试一下看这个结果是否正确)

    37a2e6039ba74123aa8dc309ae670546.png

    🍏 五、求斐波那契数列第N项

    🍻🍻斐波那契数列指的是这样一个数列:1,1,2,3,5,8,13,21,34,55,89...这个数列从第3项开始,每一项都等于前两项之和。

    🍼🍼那么,谈及递归的运算不得不提斐波那契数列这样的经典问题,下面就给大家展示一下Java语言的代码实现:

    1. public class Fib {
    2. public static int Fibo(int n){
    3. if(n<=2){
    4. return 1;
    5. }
    6. else{
    7. return Fibo(n-1)+Fibo(n-2);
    8. }
    9. }
    10. public static void main(String[] args) {
    11. Scanner sc =new Scanner(System.in);
    12. int n = sc.nextInt();
    13. System.out.println(Fibo(n));
    14. }
    15. }

    🍱🍱代码分析:虽然这种递归方法很容易理解方便使用,但是我们发现在递归的过程中,如果我们要求斐波那契数列的前40项 50项的和,以40项为例我们首先需要求 39项 38项的结果,而要得到39项的和我们要求出38的项的结果和37项的结果,进而上个38项的结果我们需要在求一次,这样发现我们有很多次的重复计算,造成了很不必要的浪费,那么我们可以通过for循环的方式来减少代码冗余度。🧂🧂

    ☕☕优化代码:

    1. public static int fib(int n) {
    2.   int last2 = 1;
    3.   int last1 = 1;
    4.   int sum = 0;
    5.   for (int i = 3; i <= n; i++) {
    6.     sum = last1 + last2;
    7.     last2 = last1;
    8.     last1 =sum;
    9.  }
    10.   return sum;
    11. }

    🍰🍰运行结果:

    51db5150b7434669ae859d398cf14d6d.png

  • 相关阅读:
    关于Redux的学习
    APS排程助动力电池企业充分利用产能
    使用Response的验证码案例_黑马javaweb案例
    【echarts】06、echarts+vue2 - 雷达图
    C#学习记录——基本图形绘制
    C#——委托
    [Leetcode]21. 合并两个有序链表
    企业知识管理怎样做?一些解决方案分享!
    无人机生态环境监测、图像处理与 GIS 数据分析
    NPDP证书有用吗?值不值得考?
  • 原文地址:https://blog.csdn.net/m0_62426532/article/details/126087220