深入浅出递归算法

news2025/1/18 6:53:59

文章目录

  • 递归思想
  • 递归的题目
    • 1.汉诺塔问题
      • 问题分析
      • 代码展示
    • 2.合并两个有序链表
      • 问题分析
      • 代码展示
    • 3.反转链表
      • 问题分析
      • 代码展示
    • 4.两两交换 链表中的节点
      • 问题分析
      • 代码展示
  • 总结

在这里插入图片描述

递归思想

递归就是将一个很大的问题拆分成子问题,然后再将子问题继续拆分,拆分成更小的子问题,最后直到不能拆分为止。
递归一共分为三个步骤,首先,我们要将一个问题拆为一些子问题,然后去看这些子问题是否有相同的方法可以继续拆分,所以递归的关键就是一个大问题是否能转换成相同类型的子问题,然后子问题是否又能继续转换成相同类型的子问题,注意这里我们就需要搞定我们这个递归的函数传递的参数具体需要什么,也就是(函数头),当我们确立了子问题之后,我们就需要进行函数体的书写了,书写函数体主要围绕单个子问题进行,因为,我们的一个大的问题都可以拆分为一个个的小的子问题,所以这些子问题都可以通过一个方法来处理,所以只需要对一个子问题进行书写函数体就行了,最后,我们需要防止无限递归,也就是递归的终止条件,向上归的过程。

总结:递归的方法

  1. 找到类型相同的子问题
  2. 对某个子问题进行函数体方法的书写
  3. 递归的出口----终止条件的判断

递归的题目

1.汉诺塔问题

问题分析

这里是引用
输入和输入:
在这里插入图片描述

在这里插入图片描述

首先我们来分析:
1.找到子问题

可以看到子问题很容易就出来了,我们不管有多少个盘子,我们都可以将上面的n-1个盘子看成一个整体,然后将上面n-1个盘子借助C移到B柱上,然后将最下面的盘子移到C上,然后再对上面n-1个盘子实行相同的方法,对上面n-1个盘子上的n-2个盘子用刚刚一样的方法。

这里子问题找到了,我们就可以确定我们的函数头和传递的参数了,对于上面的图我们传递的函数头就可以用下面类似方式写出:dfs(A,B,C,n).
2.用单个子问题寻找函数体
单个子问题是:
在这里插入图片描述
首先第一步是:dfs(A,C,B,n-1)

这句代码的意思是将A柱上n-1个盘子从A移到B上,借助C

第二部是:A.back()->C(伪代码)

将A上最后一个盘子移到C,当我进行了第一步递归之后,只剩下最后一个盘子了,所以,我需要将最后一个盘子移到C上

第三部:dfs(B,A,C,n-1)

将B柱上剩下的盘子移到C上,注意中间的过程我们不需要管,他会不断拆分,我们只需要找到同一的方法即可

3.递归出口
对于递归的出口,我们可以看上面的子问题,当我们只有一个盘子的时候我们就直降将这个盘子移到C柱上了,所以这里的最后的递归出口就是当只有一个盘子的时候。

代码展示

class Solution {
public:
    void dfs(vector<int>& x,vector<int>&y,vector<int>&z,int n)
    {
        if(n==1)//当只有一个盘子的时候移到z柱上
        {
            //移到z柱上
            z.push_back(x.back());
            //将x柱上的值删除
            x.pop_back();
            return;
        }
        //将x柱上的整体的n-1个盘子整体移到y柱上
        dfs(x,z,y,n-1);
        //然后将x柱上剩下的一个盘子移动到z柱上
        z.push_back(x.back());
        x.pop_back();
        //最后将y柱上的剩下的盘子移动到z柱上即可,借助x柱,注意:y柱上有n-1个盘子
        dfs(y,x,z,n-1);
    }
    void hanota(vector<int>& a, vector<int>& b, vector<int>& c)
    {
        dfs(a,b,c,a.size());
    }
};

2.合并两个有序链表

在这里插入图片描述
样例输入和输出:
在这里插入图片描述

问题分析

1.寻找子问题
这里其实我们可以选一个小的作为头,选好这个头之后将这个头去指向这个函数头,这个函数头就是去给我们合并的函数。
确定函数头:dfs(l1,l2)
2.根据单个子问题找到函数体

这里我们可以通过子问题找到函数体:首先函数头是传递两个链表的头,然后返回的是新的头结点,所以这里我们只需要取两个链表的中的小的为新的头,然后去链接剩下的两个链表

函数体:min->next=dfs(min->next,other)

这里大致就可以将函数体写成这样,小的链表的头指向小的链表的剩下的部分和另一个链表

3.递归出口
递归出口:当有一个函数为空时直接返回另一个链表,注意:这里其实可以这样想,当我们链接当一个链表为空的时候是不是另一个链表链接上去肯定是有序的,所以这里我们只需要返回另一个部位空的链表即可。

代码展示

class Solution 
{
public:
    ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) 
    {
        if(list1==nullptr)return list2;
        if(list2==nullptr)return list1;
        if(list1->val<=list2->val)
        {
            list1->next=mergeTwoLists(list1->next,list2);//list1作为头结点合并list1->next,和list2
            return list1;
        }
        else
        {
            list2->next=mergeTwoLists(list1,list2->next);//连接list2->next和list1
            return list2;
        }
    }
};

3.反转链表

在这里插入图片描述

问题分析

这道题我们其实也可以用递归来做,我们要将整个链表翻转其实可以看做将1后面的链表翻转,剩下的链表翻转又可以分解成剩下的链表的剩下的部分翻转,接下来我用一个图方便理解

在这里插入图片描述
大概就是上图的意思,我们先深度优先遍历到最后一个节点,然后再向上翻转注意,这里我没有志向nullptr,但是我们每次翻转的时候都要指向nullptr,这是为了递归的统一。

函数头
函数头:dfs(head)

函数体

函数体:newhead=reverseList(head->next);
我们只需要创一个新的头来等于剩下的翻转过的链表,注意:这里我们翻转过的链表是抽象的递归,具体是怎么完成的计算机会完成,我们只需要给出方法,这里我们已经得到了翻转链表的方法之后,我们就可以直接将就的head与翻转过的链表进行连接即可。

递归出口
当当前节点指向的下一个是nullptr的时候或者当当前节点是nullptr的时候就直接返回当前节点。

代码展示

class Solution 
{
public:
    ListNode* reverseList(ListNode* head) 
    {
        //出口
        if(head==nullptr||head->next==nullptr)
        {
            return head;
        }
        ListNode*newhead=reverseList(head->next);
        //原本head->next是head的next但是现在要反转过来
        //就要把head的next节点指向自己就是head->next->next=head;
        head->next->next=head;
        head->next=nullptr;
        return newhead;
    }
};

4.两两交换 链表中的节点

这里是引用

问题分析

这里这道题和上一道题其实很相似,我们其实只需要将后面所有应该交换的节点全部交换了,然后将后面节点的新的头给前面两个节点连接上即可,注意这里后面是怎么交换的我们也不知道,但是我们用一个函数去让他交换,我们让他交换前两个节点后面剩下的节点,它会转换成叫唤后面剩下的节点除了前两个节点外的剩下的节点,这样一直递归下去,直到遇到递归出口为止。

函数头
函数头:dfs(head)

函数体
注意这里我们返回的交换之后链表的新的头,意思就是当我们把除了1和2节点外的所有节点外的节点交换之后,会返回一个新的节点,注意看上面给出的示例,意思就是当我们用一个递归表示的话,返回的就是4这个节点,所以我们可以直接用tmp存储这个节点,然后将前面两个节点的指向进行变化即可。
函数体:

ListNode*tmp=swapPairs(head->next->next);
auto next=head->next;
head->next->next=head;
head->next=tmp;

递归出口
这里的递归出口还是当遇到空节点或者下一个节点是空节点直接返回当前节点

代码展示

class Solution {
public:
    ListNode* swapPairs(ListNode* head) {
        if(head==nullptr||head->next==nullptr)
        {
            return head;
        }
        ListNode*tmp=swapPairs(head->next->next);
        
        auto next=head->next;
        head->next->next=head;
        head->next=tmp;
        return next;
    }
};

总结

递归算法作为计算机科学中的一种基本思想,展现了其简洁优雅和强大的解决问题能力。从数学计算到复杂的数据结构处理,递归提供了一种自然且直观的方法来分解和解决问题。尽管递归在某些情况下可能带来性能和资源上的挑战,但通过优化技术如记忆化存储和尾递归优化,我们可以克服这些困难,实现高效的递归算法。

递归不仅仅是编程技术,更是一种思维方式。通过理解递归的本质,我们能够培养出更好的抽象思维能力,解决更复杂的计算问题。希望这篇博客能够帮助你更好地理解递归算法,并激发你在编程中更多地应用和探索这一强大的工具。

无论你是编程新手还是经验丰富的开发者,掌握递归算法都会为你提供一种新的视角,帮助你在算法和数据结构的学习和应用中取得更大的进步。让我们一起拥抱递归的美妙世界,不断挑战和提升自我!

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

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

相关文章

安全设计 | Microsoft 威胁建模工具Threat Modeling Tool安装及使用详解(文末附样例)

1. 概览 微软威胁建模工具&#xff08;Threat Modeling Tool&#xff09;是 Microsoft 安全开发生命周期 (SDL&#xff0c;Security Develop LifeCycle) 的核心要素。 当潜在安全问题处于无需花费过多成本即可相对容易解决的阶段&#xff0c;软件架构师可以使用威胁建模工具提…

C语言 | Leetcode C语言题解之第109题有序链表转换二叉搜索树

题目&#xff1a; 题解&#xff1a; int getLength(struct ListNode* head) {int ret 0;while (head ! NULL) {ret, head head->next;}return ret; }struct TreeNode* buildTree(struct ListNode** head, int left, int right) {if (left > right) {return NULL;}int …

力扣刷题---409. 最长回文串【简单】

题目描述 给定一个包含大写字母和小写字母的字符串 s &#xff0c;返回 通过这些字母构造成的 最长的回文串 。 在构造过程中&#xff0c;请注意 区分大小写 。比如 “Aa” 不能当做一个回文字符串。 示例 1: 输入:s “abccccdd” 输出:7 解释: 我们可以构造的最长的回文串…

分享:大数据风险检测报告,哪里查询比较好?

随着大数据技术的发展&#xff0c;逐渐被运用到各个领域&#xff0c;基于大数据技术的个人风险检测也就是我们常说的大数据报告在金融环境中运用的十分普遍&#xff0c;那大数据风险检测报告哪里查询比较好呢?本文就为大家简单介绍一下。 大数据风险检测报告查询能查到什么? …

Leetcode刷题笔记3

18. 四数之和 18. 四数之和 - 力扣&#xff08;LeetCode&#xff09; 给你一个由 n 个整数组成的数组 nums &#xff0c;和一个目标值 target 。请你找出并返回满足下述全部条件且不重复的四元组 [nums[a], nums[b], nums[c], nums[d]] &#xff08;若两个四元组元素一一对应&…

骨位深间距小模具镶件如何走水路?3D打印让一切简单

在模具制造领域&#xff0c;骨位深且间距小的模具镶件由于结构复杂&#xff0c;传统加工方法难以制造出符合要求的冷却水路&#xff0c;导致模具在注塑过程中容易产生热量积聚&#xff0c;进而引发烫伤、缩孔等不良。然而&#xff0c;随着3D打印技术的飞速发展&#xff0c;这些…

爬虫逆向实例小记——某数据知识管理网站-DES-ECB模式

aHR0cHM6Ly9rZC5uc2ZjLmNuL2ZpbmFsUHJvamVjdEluaXQ 注意&#xff1a;本文是逆向部分比较少&#xff0c;主要为了流程走通&#xff0c;限于代码搬运工。 第一步:分析页面 此网站经过请求响应&#xff0c;可以看出响应内容为加密内容。 第二步&#xff1a;判断加密类型 在XHR …

【Linux】从零开始认识进程间通信 —— 管道

送给大家一句话&#xff1a; 人要成长&#xff0c;必有原因&#xff0c;背后的努力与积累一定数倍于普通人。所以&#xff0c;关键还在于自己。 – 杨绛 从零开始认识进程间通信 1 为什么要进程间通信2 进程如何通信3 进程通信的常见方式4 管道4.1 什么是管道4.2 管道通信的系…

postgresql insert on conflict 不存在则插入,存在则更新

向一张表执行插入动作&#xff0c;如果插入的字段数据已存在&#xff0c;则执行更新操作&#xff0c;不存在则进行插入操作。 1、创建一张表 CREATE TABLE "user_info" ( "id" int2 NOT NULL, "name" varchar(20) COLLATE "pg_catalog&quo…

近邻算法详细

近邻算法&#xff0c;特别是K-近邻算法&#xff08;K-Nearest Neighbors, KNN&#xff09;&#xff0c;是一种基于实例的学习方法&#xff0c;广泛应用于分类和回归分析任务。下面是K-近邻算法的详细说明&#xff1a; 基本概念 K-近邻算法的核心思想是“物以类聚”&#xff0…

基于transformers框架实践Bert系列1--分类器(情感分类)

本系列用于Bert模型实践实际场景&#xff0c;分别包括分类器、命名实体识别、选择题、文本摘要等等。&#xff08;关于Bert的结构和详细这里就不做讲解&#xff0c;但了解Bert的基本结构是做实践的基础&#xff0c;因此看本系列之前&#xff0c;最好了解一下transformers和Bert…

node版本管理nvm详细教程

安装 nvm 之前先清理node相关的所有配置&#xff0c;如环境变量、.npmrc文件、node_cache、node_global 等 一、下载nvm 任选一处下载即可 官网&#xff1a;Releases coreybutler/nvm-windows (github.com) 码云&#xff1a;nvm下载仓库: nvm下载仓库 百度网盘&#xff1…

基于GA遗传优化的CNN-GRU的时间序列回归预测matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 4.1 CNN-GRU模型架构 4.2 GA优化CNN-GRU流程 5.算法完整程序工程 1.算法运行效果图预览 2.算法运行软件版本 MATLAB2022a 3.部分核心程序 ...........................................…

雷电预警监控系统:守护安全的重要防线

TH-LD1在自然界中&#xff0c;雷电是一种常见而强大的自然现象。它既有震撼人心的壮观景象&#xff0c;又潜藏着巨大的安全风险。为了有效应对雷电带来的威胁&#xff0c;雷电预警监控系统应运而生&#xff0c;成为现代社会中不可或缺的安全防护工具。 雷电预警监控系统的基本…

代码随想录算法训练营第14天 |● 理论基础 ● 递归遍历 ● 迭代遍历 ● 统一迭代

文章目录 前言二叉树的递归遍历&#x1f496;递归算法基本要素代码 迭代遍历-需要先理清思路再写前向迭代法后序迭代中序迭代 迭代法统一写法总结 前言 理论基础 需要了解 二叉树的种类&#xff0c;存储方式&#xff0c;遍历方式 以及二叉树的定义 记录我容易忘记的点 题目…

打造AI虚拟伴侣 - 优化方案

第一部分:框架优化概述 1、精确定位: 构建一个高度灵活且用户友好的平台,旨在通过无缝集成多种大型语言模型(LLMs)后端,为用户创造沉浸式的角色交互体验。不仅适配电脑端,还特别优化移动端体验,满足二次元AI虚拟伴侣市场的特定需求。 2、核心功能强化: 增强后端兼容…

大数据Hive中的UDF:自定义数据处理的利器(下)

在上一篇文章中&#xff0c;我们对第一种用户定义函数&#xff08;UDF&#xff09;进行了基础介绍。接下来&#xff0c;本文将带您深入了解剩余的两种UDF函数类型。 文章目录 1. UDAF1.1 简单UDAF1.2 通用UDAF 2. UDTF3. 总结 1. UDAF 1.1 简单UDAF 第一种方式是 Simple(简单…

叶面积指数(LAI)数据、NPP数据、GPP数据、植被覆盖度数据获取

引言 多种卫星遥感数据反演叶面积指数&#xff08;LAI&#xff09;产品是地理遥感生态网推出的生态环境类数据产品之一。产品包括2000-2009年逐8天数据&#xff0c;值域是-100-689之间&#xff0c;数据类型为32bit整型。该产品经过遥感数据获取、计算归一化植被指数、解译植被类…

测量模拟量的优选模块:新型设备M-SENS3 8

| 具有8路自由选择通道的新型设备M-SENS3 8 IPETRONIK推出的模拟量测量设备——M-SENS3 8是新一代设备的新成员。该模块具有8个通道&#xff0c;能够自由选择测量模式&#xff0c;不仅支持高精度电压和电流的测量&#xff0c;还新增了频率测量模式。各通道分辨率高达18位&…

Selenium常用命令(python版)

日升时奋斗&#xff0c;日落时自省 目录 1、Selenium 2、常见问题 1、Selenium 安装Python和配置环境没有涉及 注&#xff1a;如有侵权&#xff0c;立即删除 首先安装selenium包&#xff0c;安装方式很简单 pip install selenium 注:我这里已经安装好了&#xff0c;所以…