【Java算法】递归

news2024/9/25 9:31:09

 🔥个人主页: 中草药 

 🔥专栏:【算法工作坊】算法实战揭秘


 🍇一.递归

概念

递归是一种解决问题的方法,其中函数通过调用自身来求解问题。这种方法的关键在于识别问题是否可以被分解为若干个相似但规模更小的子问题。递归函数有两个主要组成部分:

  1. 基本情况(Base Case):这是递归调用的终止条件。每个递归函数都必须有一个或多个明确的基本情况,否则递归将无限进行下去。
  2. 递归步骤(Recursive Step):这是函数调用自身的部分。在此步骤中,问题被分解为更小的子问题,然后递归地求解这些子问题。

递归的本质可以理解为 可以把一个主问题拆分成若干个相同的子问题

宏观的理解递归

        递归其实是比较抽象的一种算法,完全熟练的运用它,需要时间的积累与深入的理解,因此我们可以学会宏观的去看待递归,帮助我们去解决算法问题

1.不必过分关注递归的细节展开图

2.把递归的函数看做一个黑盒(不去深究)

3.相信这个黑盒一定能完成这个任务

用法 

如何写好一个递归

  1. 先找到相同的子问题--->函数头的设计
  2. 只关心某一个子问题是如何解决的--->函数体的书写
  3. 注意一下递归函数的出口

应用场景

递归非常适合处理以下类型的问题:

  • 分治法:问题可以被分解为独立且较小的子问题。
  • 树形结构:例如文件系统的目录树、DOM树等。
  • 回溯算法:如八皇后问题、迷宫寻路等。

 🍈二. 面试题 08.06. 汉诺塔问题

题目链接:面试题 08.06. 汉诺塔问题

代码

public void hanota(List<Integer> A, List<Integer> B, List<Integer> C) {
       dfs(A,B,C,A.size());
       return;
    }

    public void dfs(List<Integer> A, List<Integer> B, List<Integer> C,int n){
        //算法的出口
        if(n==1){
            C.add(A.remove(A.size()-1));
            return;
        }

        dfs(A, C, B, n-1);
        C.add(A.remove(A.size()-1));
        dfs(B, A, C, n-1);
    }

算法原理 

汉诺塔问题是递归问题的一个经典问题,和常规问题一样,拆分成细小的步骤

汉诺塔问题通常有三个柱子A、B、C,以及n个不同大小的圆盘,初始时所有的圆盘都堆叠在柱子A上,要求将它们全部移动到柱子C上,但在移动过程中必须遵守以下规则:

  1. 每次只能移动一个圆盘。
  2. 在任何时刻,一个较大圆盘都不能放在较小圆盘上面

代码详解

  • hanota 方法是主入口点,它接受三个列表作为参数,分别代表三个柱子A、B、C。
  • dfs 方法是一个递归函数,它接受四个参数:柱子A、B、C以及当前要移动的圆盘数量n。
  • n == 1时,这是递归的基本情况,直接将A上的最后一个圆盘移动到C。
  • 对于n > 1的情况,首先递归地将n-1个圆盘从A借助B移动到C;然后将A上的最后一个圆盘直接移动到C;最后再递归地将n-1个圆盘从B借助A移动到C。

🍉 三. 21.合并两个有序链表

题目链接:21.合并两个有序链表

代码

public ListNode mergeTwoLists(ListNode list1, ListNode list2) {
        //不能用else
        if(list1==null){
            return list2;
        }
        if(list2==null){
            return list1;
        }

        if(list1.val<=list2.val){
            list1.next=mergeTwoLists(list1.next,list2);
            return list1;
        }else{
            list2.next=mergeTwoLists(list1,list2.next);
            return list2;
        }
    }

算法原理

  1. 特殊情况处理:首先检查输入的两个链表list1list2是否为空。
    • 如果list1为空,则直接返回list2
    • 如果list2为空,则直接返回list1
  2. 递归合并:如果两个链表都不为空,则比较当前节点的值。
    • 如果list1的当前节点值小于等于list2的当前节点值,那么将list1next指向list1.nextlist2的合并结果,并返回list1
    • 否则,将list2next指向list1list2.next的合并结果,并返回list2

这段代码使用了递归的方式来合并两个有序链表。其基本思想是:

  • 基本情况:如果其中一个链表为空,那么合并的结果就是另一个链表。
  • 递归步骤:如果两个链表都不为空,则比较当前节点的值。
    • 如果list1的当前节点值小于等于list2的当前节点值,那么list1将成为合并后链表的一部分,然后递归地去合并list1的下一个节点和list2
    • 如果list2的当前节点值小于list1的当前节点值,那么list2将成为合并后链表的一部分,然后递归地去合并list1list2的下一个节点。

🍊四. 206.反转链表

代码

public ListNode reverseList(ListNode head) {
        if(head==null||head.next==null){
            return head;
        }
        ListNode newHead=reverseList(head.next);
        //逆置
        head.next.next=head;
        head.next=null;

        return newHead;
    }

算法原理

这里可以比较这道题的 迭代代码 加深理解

public ListNode reverseList1(ListNode head) {
        if(head==null){
            return head;
        }
        ListNode cur=head.next;
        head.next=null;
        while(cur!=null){
            ListNode curn=cur.next;
            cur.next=head;
            head=cur;
            cur=curn;
        }
        return head;
    }
  1. 基本情况:如果当前节点head为空或者当前节点是链表的最后一个节点(即head.next == null),那么直接返回head。这是因为如果链表为空或者只剩一个节点,就不需要反转了,直接返回即可。

  2. 递归步骤:如果当前节点不是最后一个节点,递归地反转head.next,即反转当前节点之后的所有节点。这里newHead将是反转后的新头节点,即原来链表的最后一个节点成为了新的头节点。 

  3. 链接反转:在递归调用返回后,将当前节点head连接到新头节点newHead后面。这一步是通过将head.next.next = head;来实现的,即将head插入到新反转链表的头部。

    • 注意,这里的head.next是原来链表中的下一个节点,现在它已经是反转链表的头节点了,所以我们通过head.next.next来访问原来链表的下下个节点,现在变成了反转链表的第二个节点。
  4. 断开连接:将headnext设为null,这样原来的链表就被切断了,从而完成了反转。

🍋五.  24.两两交换链表中的节点

题目链接:24.两两交换链表中的节点

代码

 public ListNode swapPairs(ListNode head) {
        if(head==null||head.next==null){
            return head;
        }
        ListNode tmp=swapPairs(head.next.next);
        ListNode ret=head.next;
        ret.next=head;
        head.next=tmp;

        return ret;
    }

算法原理

此代码,建议先用一个小链表模拟实现以下该过程,帮助理解这个递归

基本情况:如果当前节点head为空或head.next为空(即链表为空或只有一个节点),则直接返回head。这是因为单个节点或空链表无需交换。

递归步骤

  • 首先递归地调用swapPairs(head.next.next),这意味着我们假设head.next.next之后的部分已经被正确地交换好了。递归的目的是处理当前节点之后的所有节点。
  • tmp变量存储了从head.next.next开始的交换好的链表部分。
  • ret变量存储了head.next,即当前节点的下一个节点,它将成为新的头节点。
  • 接下来,将ret.next设置为head,这样就实现了headhead.next这两个节点的交换。
  • 最后,将head.next设置为tmp,这样就将剩余部分的链表正确地连接到了交换后的两个节点之后。

 🍌六.  50.Pow(x,n)

题目链接:50.Pow(x,n)

代码

public double myPow(double x, int n) {
        //存在负数情况,若为负数变为倒数
        return n<0?1/pow(x,-n):pow(x,n);
    }

    public double pow(double x,int n){
        if(n==0){
            return 1;
            //相当于x的0次方幂
        }
        double tmp=pow(x,n/2);
        //若为奇数,还需要在 * 上一个x
        return n%2==0?tmp*tmp:tmp*tmp*x;
    }

算法原理 

当拿到这道题,大多数人首先会想到循环,但问题是常规的这种循环会超时,因此我们应该将问题用递归的方式去简化,拆分成相同的子问题

这个算法采用了分治的思想,通过递归地将问题规模减半来快速计算幂。

  1. 基本情况:当n为0时,任何数的0次幂都是1。
  2. 递归步骤
    • 如果n是偶数,那么x^n可以表示为(x^(n/2))^2
    • 如果n是奇数,那么x^n可以表示为x * (x^(n-1))

通过递归地计算x^(n/2),我们可以将问题规模减半,从而大大减少了计算次数。这种方法的时间复杂度大约是O(log n),相比直接循环相乘的O(n)复杂度要高效得多。

 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

深入Redis:复杂的集群

广义的集群&#xff0c;可能说只要是多台机器组成了分布式系统&#xff0c;就可以称之为集群。 狭义的集群&#xff0c;指的是Redis提供的集群模式&#xff0c;这个集群模式之下&#xff0c;主要是解决存储空间不足的问题&#xff0c;以及如何拓展存储空间。 之前的哨兵模式&…

C++中string的简单实现

string的简单实现中一些函数的实现可以复用一些其他的函数来实现&#xff1b;比较重要的是在实现是深浅拷贝问题 目录 string的结构 实现构造和析构 reserve扩容 容量 push_back和append insert和erase的实现 swap的实现&#xff08;不是成员函数但是string类的友元&…

【c++】类和对象详解

✅博客主页:爆打维c-CSDN博客​​​​​​ &#x1f43e; &#x1f539;分享c语言知识及代码 来都来了! 点个赞给博主个支持再走吧~&#xff01; 一.类的定义 &#xff08;1&#xff09;类定义格式 class为类定义的关键字&#xff0c;定义一个类格式如下: class 类名{//代码…

kubelet 探针

目录 1 k8s中kubelet 探针的介绍 1.1 探针是由 kubelet 对容器执行的定期诊断: 1.2 Kubelet 可以选择是否执行在容器上运行的三种探针执行和做出反应&#xff1a; 1.3 ReadinessProbe 与 LivenessProbe 的区别 1.4 StartupProbe 与 ReadinessProbe、LivenessProbe 的区别 2 实…

CCRC-DSA数据安全评估师:网络安全风险评估

1.网络安全风险评估概述 1.1概念 在当今信息化时代&#xff0c;网络安全成为了组织不可或缺的一部分。 风险评估作为一种科学方法&#xff0c;其目的是对网络系统的保密性、完整性、可控性和可用性这四个核心安全属性进行深入分析。 这一过程不仅包括识别网络系统中存在的脆…

Snipaste无法使用F1、F3等快捷键的保姆级解决方法

在Snipaste中按F1、F3等快捷键无效的可能原因&#xff1a; 1. 软件设置&#xff1a; 检查Snipaste的设置&#xff0c;确保F1被正确设置为截屏热键&#xff0c;并确认没有其他软件占用或冲突。 2. 热键冲突&#xff1a; 笔记本电脑的功能键&#xff08;F1-F12&#xff09;通常…

MySQL 数据库:原理、应用与发展

摘要&#xff1a;本文深入探讨了 MySQL 数据库相关内容。首先介绍了 MySQL 作为开源关系型数据库管理系统的显著特点&#xff0c;包括易用性、跨平台性、高性能、可扩展性、开源免费以及数据安全性等方面。接着详细阐述了其安装与配置过程&#xff0c;涵盖在不同操作系统上的安…

STM32使用 :串口的接收与发送

一、串口 在 STM32 中&#xff0c;串口&#xff08;UART&#xff0c;通用异步收发传输器&#xff09;是用于串行通信的外设。它在嵌入式系统中的作用非常广泛&#xff0c;主要包括几个方面 数据通信 串口用于微控制器与其他设备之间的数据传输。这些设备可以是其他微控制器、…

F12抓包08:查看网站Cookie

课程大纲 1、查看Cookie 1. 应用界面查看&#xff1a;按F12进入浏览器的开发者模式 - “应用”&#xff08;Application&#xff09; - Cookie&#xff0c;可查看Cookie并进行增、删、改、查操作。 2. 控制台命令行查看&#xff1a;按F12进入浏览器的开发者模式 - “控制台”&…

【无标题】nginx服务器代码信息、数据库连接信息、敏感文件的路径、服务器版本信息发起有针对性的攻击

Nginx敏感文件的路径、服务器版本信息 Nginx 403、404、500等错误时&#xff0c;返回详细错误信息。报错信息中可能会包含服务器代码信息、数据库连接信息、敏感文件的路径、服务器版本信息等&#xff0c;攻击者可以利用这些信息来寻找已知的漏洞&#xff0c;从而发起有针对性…

『功能项目』管理器基类【38】

我们打开上一篇37单例模式框架的项目&#xff0c; 本章要做的事情是编写管理器基类 首先创建脚本&#xff1a;ManagerBase.cs using UnityEngine; public abstract class ManagerBase : MonoBehaviour{public virtual void Init() { } } public class ManagerBase<T> : …

【Qt系列样式表】探索Qt Widget的艺术化设计与应用(Macos风格)(持续更新中...)

✨✨ Rqtz 个人主页 : 点击✨✨ &#x1f308;Qt系列专栏:点击 &#x1f388;PyQt系列专栏:点击&#x1f388; &#x1f388;Qt智能车上位机专栏: 点击&#x1f388; &#x1f388;Qt串口助手专栏:点击&#x1f388; &#x1f4ab;宗旨:共享IT之美,共创机器未来 目录 界面…

毕业论文word页眉页脚和页码的问题

多找B站视频查看视频操作方法。 问题1&#xff1a; 不同章节的页眉显示不同内容。 解决办法&#xff1a; “布局”里面插入分节符 &#xff08;非 分页符。&#xff09; 问题2&#xff1a; 页眉需要奇偶页内容不同 解决办法&#xff0c;编辑页眉部分&#xff0c;设置 奇偶页…

Qt (15)【Qt窗口 —— 字体对话框 QFontDialog | 输入对话框 QInputDialog】

阅读导航 引言一、字体对话框 QFontDialog1. 简介2. 基本用法3. 示例代码 二、输入对话框 QInputDialog1. 简介2. 基本用法&#xff08;1&#xff09;双精度浮点型输入数据对话框&#xff08;2&#xff09;整型输入数据对话框&#xff08;3&#xff09;选择条目型输入数据框 3.…

李沐对大模型趋势的几点判断

李沐是上海交通大学 2011 届计算机科学与工程系本硕系友。他曾担任亚马逊资深首席科学家&#xff0c;加州大学伯克利分校和斯坦福大学的访问助理教授&#xff0c;是前 Marianas Labs 联合创始人&#xff0c;深度学习框架 Apache MXNet 的创始人之一。目前是 BosonAI 联合创始人…

【数据分析】标准误差与标准差的区别

标准误差&#xff08;Standard Error, SE&#xff09;和标准差&#xff08;Standard Deviation, SD&#xff09;是两个在统计学中非常重要的概念&#xff0c;但它们的含义和用途有所不同。以下是它们之间的主要区别&#xff1a; 定义&#xff1a; 标准差&#xff1a;衡量单个数…

【mybatis】使用模糊查询时报错:Encountered unexpected token: “?“ “?“

报错信息如下&#xff1a; Mapper.xml报错代码&#xff1a; AND HILIST_NAME like %#{hilistName}% 解决方案&#xff1a; 把模糊查询的 sql 语句改为使用 CONCAT 命令拼接, 就不会报错了。 AND HILIST_NAME like CONCAT(%, #{hilistName},%)

深入探讨-JavaScript-逻辑赋值运算符

null && 100**2 // null undefined && 100**2 // undefined 逻辑赋值运算符 && || ?? 这个运算符将赋值与条件逻辑运算符结合在一起&#xff0c;因此命名为**“逻辑赋值”** 。它们只是一种简写&#xff0c; 例如&#xff0c;x && y是x …

Windows本地部署ollama并实现无公网IP远程运行qwen大语言模型

文章目录 前言1. 运行Ollama2. 安装Open WebUI2.1 在Windows系统安装Docker2.2 使用Docker部署Open WebUI 3. 安装内网穿透工具4. 创建固定公网地址 前言 本文主要介绍如何在Windows系统快速部署Ollama开源大语言模型运行工具&#xff0c;并安装Open WebUI结合cpolar内网穿透软…

打造智能数据分析平台:基于 Flask 的数据处理与模型精度验证系统

数据分析和机器学习已成为企业和科研中不可或缺的核心技术。在这个数据驱动的时代&#xff0c;能够快速处理海量数据&#xff0c;并通过智能算法提取出有用信息&#xff0c;成为了提升竞争力的关键。为了解决这些需求&#xff0c;我基于 Flask 开发了一款功能强大、模块化的数据…