【数据结构算法(一)】递归篇(常见实例讲解)

🌈键盘敲烂,年薪30万🌈

本篇讲解实例:

  • 斐波那契、兔子问题、猴子吃桃问题、跳台阶问题、汉诺塔、杨辉三角

用到的递归思想:

  • 无记忆递归、记忆递归(重点掌握)

目录

一、斐波那契:

①无记忆多路递归:

②⭐记忆递归:

二、兔子问题:

三、跳台阶问题:

四、汉诺塔问题:

五:杨辉三角问题:

①无记忆递归:

②⭐记忆递归:

六、猴子吃桃问题:


一、斐波那契:

问题描述:

这个数列的每个数字都是前两个数字之和,数列的第一个和第二个数规定为1

①无记忆多路递归:
  • 时间复杂度:O(n^2) -  很恐怖
public class FibonaciNoMemory {
    // 1 1 2 3 5 8 13 21 34 55……
    public static void main(String[] args) {
        int n = 10;
        //无记忆性的递归
        int ans2 = noMemoryRecursion(n);
        System.out.println(ans2);

    }

    private static int noMemoryRecursion(int n) {
        if(n == 1 || n == 2){
            return 1;
        }
        return noMemoryRecursion(n-1) + noMemoryRecursion(n-2);

    }
}
②⭐记忆递归:
  • 时间复杂度:O(n)
public class FibonaciRemind {
    public static void main(String[] args) {
        int n = 10;
        int ans = remindRecursion(n);
        System.out.println(ans);
    }
    private static int remindRecursion(int n) {
        int[] cache = new int[n+1];
        Arrays.fill(cache, -1);
        cache[0] = 1; cache[1] = 1;
        return help(n-1, cache);
    }

    private static int help(int n, int[] cache) {
        if(cache[n] != -1){
            return cache[n];
        }
        int val = help(n-1, cache) + help(n-2, cache);
        cache[n] = val;
        return val;
    }
}

 

二、兔子问题:

问题描述:

有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第三个月后每个月又生一对兔子,假如兔子都不死,问每个月的兔子总数为多少?

代码同斐波那契差不多,多了个求和,这个兔子问题就是列昂纳多·斐波那契引申出的。

public class a06_rabbit {
    public static void main(String[] args) {
        int month = 10;
        int count = getCount(month);
        System.out.printf("第十个月,共%d只兔子", count);
    }

    private static int getCount(int month) {
        int[] cache = new int[month];

        cache[0] =  1;cache[1] = 1;

        help(month-1, cache);
        int total = 1;
        for (int i = 0; i < month; i++) {
            total += cache[i];
        }
        return total;
    }

    private static int help(int month, int[] cache) {
        if(cache[month] != 0){
            return cache[month];
        }
        cache[month] = help(month - 1, cache) + help(month - 2, cache);
        return cache[month];
    }
}

 

三、跳台阶问题:

问题描述:

鸡哥跳台阶,有时跳一阶,有时跳二阶,问,若有10层台阶,有多少种跳法

public class SkipStairs {
    public static void main(String[] args) {
        int n = 10;
        int ans = getCount(n);
        System.out.printf("共有%d种跳法", ans);
    }

    private static int getCount(int n) {
        return help(n);
    }

    private static int help(int n) {
        if(n == 1){
            return 1;
        }
        if(n == 2){
            return 2;
        }
        return help(n-1) + help(n-2);

    }
}

 

四、汉诺塔问题:

问题描述:

有三根柱子,编号为A、B、C,开始时在柱子A上有一些个圆盘,它们按照从下到上的顺序递增(最下面的最大,最上面的最小)。现在要将这些圆盘从柱子A移动到柱子C,中间可以借助柱子B,但有一些规则需要遵守:

  1. 每次只能移动一个圆盘。
  2. 移动过程中,大圆盘不能放在小圆盘上面。
public class Demo1 {
    static LinkedList<Integer> a = new LinkedList<>();
    static LinkedList<Integer> b = new LinkedList<>();
    static LinkedList<Integer> c = new LinkedList<>();
    public static void main(String[] args) {
        a.addLast(3);
        a.addLast(2);
        a.addLast(1);
        move(3, a, b, c);

    }
    private static void move(int n, LinkedList<Integer> a, LinkedList<Integer> b, LinkedList<Integer> c) {
        if(n == 0){
            return;
        }
        //转移n-1个到b - 要借助c
        move(n-1, a, c, b);
        //将最大的移到C
        c.add(a.removeLast());
        myPrint();
        //将n-1个到c - 要借助a
        move(n-1, b, a, c);
    }
    private static void myPrint() {
        System.out.println(a);
        System.out.println(b);
        System.out.println(c);
        System.out.println("===============");
    }
}

 

五:杨辉三角问题:

问题描述:有个三角形,每一行的该数等于上一行同列数+上一行前一列的数

①无记忆递归:
public class Demo2 {
    public static void main(String[] args) {
        int n = 6;
        print(n);
    }

    private static void printSpace(int n){
        for (int i = 0; i < n; i++) {
            System.out.print(" ");
        }
    }

    private static void print(int n) {
        for (int i = 0; i < n; i++) {
            printSpace((n-i-1)*2);
            for (int j = 0; j <= i; j++) {
                System.out.printf("%-4d", getElement(i, j));
            }
            System.out.println();
        }
    }

    private static int getElement(int row, int col){
        if(col == 0 || col == row){
            return 1;
        }
        return getElement(row-1, col-1) + getElement(row-1, col);

    }
}
②⭐记忆递归:
public class Demo1 {
    public static void main(String[] args) {
        int n = 6;
        print(n);
    }

    private static void printSpace(int n){
        for (int i = 0; i < n; i++) {
            System.out.print(" ");
        }
    }

    private static void print(int n) {
        int[][] cache = new int[n][];
        for (int i = 0; i < n; i++) {
            printSpace((n-i-1)*2);
            cache[i] = new int[i+1];
            for (int j = 0; j <= i; j++) {
                System.out.printf("%-4d", getElement(cache, i, j));
            }
            System.out.println();
        }
    }

    private static int getElement(int[][] cache, int row, int col){
        if(cache[row][col] > 0){
            return cache[row][col];
        }

        if(col == 0 || col == row){
            cache[row][col] = 1;
            return 1;
        }
        cache[row][col] = getElement(cache, row-1, col-1) + getElement(cache, row-1, col);
        return cache[row][col];

    }
}

 

六、猴子吃桃问题:

问题描述:

有一只猴子摘了一堆桃子,第一天它吃了其中的一半,并再多吃了一个;第二天它又吃了剩下的桃子的一半,并再多吃了一个;以后每天都吃了前一天剩下的一半并再多吃了一个。到第n天想再吃时,发现只剩下一个桃子。问这堆桃子原来有多少个?

public class MonkeyEatPeach {

    public static void main(String[] args) {
        int days = 9; // 假设猴子在第9天时发现只剩下一个桃子

        // 调用计算桃子数量的方法
        int result = calculatePeaches(days);

        // 输出结果
        System.out.println("猴子摘的桃子总数为:" + result);
    }

    // 计算桃子数量的方法
    public static int calculatePeaches(int days) {
        if(days == 1){
            return 1;
        }
        return (calculatePeaches(days - 1) + 1) * 2;
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/164189.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

重生奇迹mu转职任务详解

重生奇迹mu神骑士怎么转 神骑士是一种转职类型&#xff0c;需要你的角色达到一定等级以及完成相应任务方可转职。以下是神骑士转职的具体步骤&#xff1a; 1.等级要求&#xff1a;首先&#xff0c;你的角色需要达到150级才能进行神骑士转职任务。 2.神骑士转职任务&#xff…

十七、Linux的组管理

1、Linux组基本介绍 在linux中的每个用户必须属于一个组&#xff0c;不能独立于组外。在linux中每个文件所有者、所在组、其它组的概念 1.所有者 2.所在组 3.其他组 4.改变用户所在的组 2、文件/目录 所有者 一般为文件的创建者&#xff0c;谁创建了该文件&#xff0c;就自…

卷积、卷积图像操作和卷积神经网络

好多内容直接看书确实很难坚持&#xff0c;就比如这个卷积&#xff0c;书上的一大堆公式和图表直接把人劝退&#xff0c;我觉得一般的学习流程应该是自顶向下&#xff0c;先整体后局部&#xff0c;先把握大概再推敲细节的&#xff0c;上来就事无巨细地展示对初学者来说很痛苦。…

【机器学习12】集成学习

1 集成学习分类 1.1 Boosting 训练基分类器时采用串行的方式&#xff0c; 各个基分类器之间有依赖。每一层在训练的时候&#xff0c; 对前一层基分类器分错的样本&#xff0c; 给予更高的权重。 测试时&#xff0c; 根据各层分类器的结果的加权得到最终结果。 1.2 Bagging …

Linux | 信号

目录 前言 一、信号基础概念 1、生活中的信号 2、Linux中的信号 二、信号的产生 1、接口介绍 2、信号产生的方式 &#xff08;1&#xff09;终端按键的方式产生信号 &#xff08;2&#xff09;系统调用接口 a、kill b、raise c、abort &#xff08;3&#xff09…

【LeetCode刷题-滑动窗口】--992.K个不同整数的子数组

992.K个不同整数的子数组 思路&#xff1a; class Solution {public int subarraysWithKDistinct(int[] nums, int k) {return atMostKDistinct(nums,k) - atMostKDistinct(nums,k-1);}//最多包含K个不同整数的子区间个数private int atMostKDistinct(int[] a,int k){int len …

【MATLAB源码-第83期】基于matlab的MIMO中V-BALST结构ZF和MMSE检测算法性能误码率对比。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 在多输入多输出&#xff08;MIMO&#xff09;通信系统中&#xff0c;V-BLAST&#xff08;垂直波束形成层间空间时间编码技术&#xff09;是一种流行的技术&#xff0c;用于提高无线通信的数据传输速率和容量。它通过在不同的…

PS 颜色取样器标尺工具 基本使用讲解

上文 PS 吸管工具基本使用方法 我们讲完了 吸管工具 那么 我们继续 打开ps先 接着 我们选择这个 颜色取样器工具 选择之后 我们鼠标在图像上随便点一下 就会出现一个标记 然后 我们可以点多几个地方 边上的信息面板就会输出 点1 和 点2 甚至 多个 点3 点4 的 颜色 RGB代码 …

Python学习(一)基础语法

文章目录 1. 入门1.1 解释器的作用1.2 下载1.3 基础语法输入输出语法与引号注释&#xff1a;变量&#xff1a; 数据类型与四则运算数据类型四则运算数据类型的查看type()数据类型的转换int()、int()、float() 流程控制格式化输出循环与遍历逻辑运算符list遍历字典dict遍历 跳出…

JavaspringbootMYSQL基于移动端的团购网站26449-计算机毕业设计项目选题推荐(附源码)

目 录 摘要 1 绪论 1.1 选题背景 1.2选题目的及意义 1.3springboot框架介绍 2 基于移动端的团购网站系统分析 2.1 可行性分析 2.2 系统流程分析 2.2.1 数据流程 3.3.2 业务流程 2.3 系统功能分析 2.3.1 功能性分析 2.3.2 非功能性分析 2.4 系统用例分析 2.5本章…

labelimg报错IndexError: list index out of range

labelimg报错IndexError: list index out of range 问题&#xff1a;标签顺序不对&#xff0c;修改classes.txt文件。每次重新打开labelimg就会重置classes.txt文件&#xff0c;同时其中不正确的标签顺序&#xff0c;会导致所画的框图范围超出图片大小而报错&#xff0c;因此也…

基于吉萨金字塔建造算法优化概率神经网络PNN的分类预测 - 附代码

基于吉萨金字塔建造算法优化概率神经网络PNN的分类预测 - 附代码 文章目录 基于吉萨金字塔建造算法优化概率神经网络PNN的分类预测 - 附代码1.PNN网络概述2.变压器故障诊街系统相关背景2.1 模型建立 3.基于吉萨金字塔建造优化的PNN网络5.测试结果6.参考文献7.Matlab代码 摘要&a…

维修一款20年前的电容测试表VC6013

一、大概情况 在咸鱼市场淘了一台VC6013电感测试表&#xff0c;本来想捡漏的&#xff0c;结果发现是一个大坑&#xff0c;不但被人维修过&#xff0c;还发现被拆了一些ic&#xff0c;网络上也找不到合适的图纸&#xff0c;只找到一份比较接近的图纸&#xff0c;但是比较下来还是…

让你彻底学会HBase

让你彻底学会HBase Apache HBase&#xff08;Hadoop DataBase&#xff09;是一个开源的、高可靠性、高性能、面向列&#xff08;这里指列族&#xff0c;非列式存储&#xff09;、可伸缩、实时读写的分布式数据库。利用 Hadoop HDFS 作为其文件存储系统&#xff0c;利用 ZooKee…

复杂类型,查询--学习笔记

1&#xff0c;复杂类型 解决问题&#xff1a;一些不容易获取到的数据&#xff0c;例如数组类型&#xff0c;集合类型等&#xff0c;获取他们的数据 -- 1.创建表 create table tb_array_person(name string,city_array array<string> )row format delimited fields term…

java 实现串口通讯

1、引入依赖 <dependency><groupId>org.scream3r</groupId><artifactId>jssc</artifactId><version>2.8.0</version> </dependency>2、配置启动串口 Component public class ContextHolder implements ApplicationContextAw…

Jave 定时任务:使用Timer类执行定时任务为何会发生任务阻塞?如何解决?

IDE&#xff1a;IntelliJ IDEA 2022.2.3 x64 操作系统&#xff1a;win10 x64 位 家庭版 JDK: 1.8 文章目录 一、Timer类是什么&#xff1f;二、Timer类主要由哪些部分组成&#xff1f;1.TaskQueue2. TimerThread 三、示例代码分析四、自定义TimerTask为什么会发生任务相互阻塞的…

Actor对象的引用 怎么设置他的材质?或设置是否启用重力?

这个蓝图我是想当重叠触发,将另一个Target Actor(一个球体)设置他的z增加50,但是为什么在触发的时候会抽搐?而且我想要设置他的材质等等这些属性都不行

SQL INSERT INTO 语句详解:插入新记录、多行插入和自增字段

SQL INSERT INTO 语句用于在表中插入新记录。 INSERT INTO 语法 可以以两种方式编写INSERT INTO语句&#xff1a; 指定要插入的列名和值&#xff1a; INSERT INTO 表名 (列1, 列2, 列3, ...) VALUES (值1, 值2, 值3, ...);如果要为表的所有列添加值&#xff0c;则无需在SQL…

【Linux】-进程间通信-匿名管道通信(以及模拟一个进程池)

&#x1f496;作者&#xff1a;小树苗渴望变成参天大树&#x1f388; &#x1f389;作者宣言&#xff1a;认真写好每一篇博客&#x1f4a4; &#x1f38a;作者gitee:gitee✨ &#x1f49e;作者专栏&#xff1a;C语言,数据结构初阶,Linux,C 动态规划算法&#x1f384; 如 果 你 …