深入浅出递归算法

文章目录

  • 递归思想
  • 递归的题目
    • 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;
    }
};

总结

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

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

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

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

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

相关文章

力扣226. 翻转二叉树(DFS的两种思路)

Problem: 226. 翻转二叉树 文章目录 题目描述思路复杂度Code 题目描述 思路 涉及二叉树的递归解法时往往需要考虑两种思路&#xff1a; 1.在递归遍历时执行题目需要的具体要求&#xff1b; 2.将一个大问题分解为多个小子问题 具体到本体&#xff1a; 思路1&#xff1a;遍历 先…

前端请求超时截断,axios timeout设置未生效情况记录

问题描述 前端请求超时截断&#xff0c;axios timeout设置未生效情况记录 timeout设置方式&#xff1a; 表现&#xff08;前端超过5min报错500&#xff0c;直接访问接口超过5min能够正常响应&#xff09;&#xff1a; 问题原因 上面的配置设置时间为1000min&#xff0c;明显…

多项式重构的平滑和法线估计-------PCL

多项式重构的平滑和法线估计 /// <summary> /// 多项式重构的平滑和法线估计 /// </summary> /// <param name"cloud"></param> /// <returns>输出一个包含平滑后的点云数据以及相应法线信息的数据结构</returns> pcl::PointCl…

ROCm上情感分析:使用循环神经网络

15.2. 情感分析&#xff1a;使用循环神经网络 — 动手学深度学习 2.0.0 documentation (d2l.ai) 代码 import torch from torch import nn from d2l import torch as d2lbatch_size 64 train_iter, test_iter, vocab d2l.load_data_imdb(batch_size)class BiRNN(nn.Module):…

sqlserver的查询(三)

目录 10. group by(分组) 11. having(对分组后的信息过滤) 可能从这里开始&#xff0c;执行顺序越来越显得重要了&#xff01;&#xff01;&#xff01; 10. group by(分组) 这个查询相比前面会有一些困难&#xff1b; 格式&#xff1a;group by 字段的集合&#xff1b; 功…

GD32F103RCT6/GD32F303RCT6-UCOSIII底层移植(4)消息队列实验

本文章基于兆易创新GD32 MCU所提供的2.2.4版本库函数开发 后续项目主要在下面该专栏中发布&#xff1a; 手把手教你嵌入式国产化_不及你的温柔的博客-CSDN博客 感兴趣的点个关注收藏一下吧! 电机驱动开发可以跳转&#xff1a; 手把手教你嵌入式国产化-实战项目-无刷电机驱动&am…

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

题目&#xff1a; 题解&#xff1a; class Solution {ListNode globalHead;public TreeNode sortedListToBST(ListNode head) {globalHead head;int length getLength(head);return buildTree(0, length - 1);}public int getLength(ListNode head) {int ret 0;while (head…

kimi :系统框架 实力学习

学海无涯&#xff0c;你&#xff0c;准备好了吗&#xff1f; 学习一个新的嵌入式系统架构&#xff0c;你"只"需要 - 1 - 手册/速查函数&#xff08;对于比较大的架构&#xff0c;F12往往返回多个结果&#xff0c;增加混乱&#xff09;&#xff1b; 2 - 源代码和VS&am…

20.有序性与内存屏障

文章目录 有序性与内存屏障1.重排序1.1.编译器重排序1.2.CPU重排序1.2.1.指令级重排序1.2.2.内存系统重排序1.3.As-if-Serial规则 2.内存屏障2.1.硬件层面的内存屏障2.1.2.写屏障2.1.3.读屏障2.1.4.全屏障 2.2.硬件层的内存屏障作用2.3.案例 有序性与内存屏障 有序性 与 可见性…

混合组网VS传统网络:智能硬件混合组网优劣势浅要解析

智能硬件混合组网是一种利用多种通信技术相结合的方法&#xff0c;以实现更灵活、更可靠的网络连接。通过蓝牙、Wi-Fi、LoRa、4G相互之间的不同通讯方式&#xff0c;根据应用场景的不同以及现场实际环境&#xff0c;优选最佳物联网混合组网方案&#xff0c;以达到部署最便捷性价…

云曦2024年春季学期期中考复现

目录 Web Web_SINGIN 简简单单的文件上传 好玩的PHP 渗透的本质 简简单单的sql re baby_re easy xor Crypto easy_rsa Rsa2 Crypto_Singin Pwn pwn_Sing Misc easy_singin Xjpg 流量分析1 流量分析3 流量分析2 Web Web_SINGIN 1.使用右键检查&#xff0c…

IMU内参标定(理论)

1、内参标定标定什么&#xff1f; 生产零偏、标度因数误差、安装误差 2、现象是什么&#xff1f; 零偏现象&#xff1a;即使没有任何运动或旋转&#xff0c;IMU传感器仍然会输出一个非零的信号。零偏是一个恒定的误差&#xff0c;导致测量值始终偏离实际值。对于加速度计&am…

DolphinDB 携手九鞅科技,助力固收投研效能飞跃

随着金融市场开放的广度与深度不断拓宽&#xff0c;金融产品呈现出多样化的发展态势&#xff0c;其中债券投资组合凭借其低风险性、高流动性与稳健的收益表现&#xff0c;逐渐成为投资理财领域备受瞩目的焦点。投资经理不仅需要了解哪些债券值得投资&#xff0c;更要对债券投资…

【GESP试卷】2024年03月Scratch四级试卷

2024年GESP03月认证Scratch四级试卷 分数&#xff1a;100 题数&#xff1a;27 一、单选题(共15题&#xff0c;每题2分&#xff0c;共30分) 010203040506070809101112131415CDBBACBCDCDADBA 1、小杨的父母最近刚刚给他买了一块华为手表&#xff0c;他说手表上跑的是鸿蒙&…

朋友正确交往方式,以及保留有效沟通,才是对朋友的尊重!

人生就像一列火车&#xff0c;从生命之初驶向生命的终点&#xff0c;路途上有很多站点&#xff0c;每一个站点都会遇到不同的人&#xff0c;结交各式各样的朋友&#xff0c;中间有人下车&#xff0c;有人上车&#xff0c;有人与你走着走着就散了&#xff0c;有人偶有相见却已是…

Qt 科目一考试系统(有源码)

项目源码和资源&#xff1a;科目一考试系统: qt实现科目一考试系统 一.项目概述 该项目是一个基于Qt框架开发的在线考试系统&#xff0c;主要实现了考试题目的随机抽取、考试时间限制、成绩统计等功能。用户可以通过界面操作进行考试&#xff0c;并查看自己的考试成绩。 二.技…

计算机网络之应用层知识点总结

6.1 网络应用模型 &#xff08;1&#xff09;应用层概述 &#xff08;2&#xff09;网络应用模型的介绍 客户/服务器&#xff08;C/S&#xff09;模型 P2P模型 6.2 域名解析系统DNS &#xff08;1&#xff09;DNS系统介绍 &#xff08;2&#xff09;域名 &#xff08;3&#…

AI爆文写作:标题需要什么?情绪炸裂,态度要激烈,行为要夸张!

现在这个传播环境下&#xff0c;在公域中&#xff0c;轻声细语&#xff0c;慢慢的说&#xff0c;无法吸引到注意&#xff0c;没有人搭理。 标题要需要情绪张扬&#xff0c;态度激烈&#xff0c;行为夸张&#xff0c;大声喧闹。 唐韧的用户群是互联网产品经理&#xff0c;阅读量…

小猫咪的奇幻冒险:一个简单的Python小游戏

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、游戏简介与演示 二、游戏开发与运行 1. 环境搭建 2. 代码解析 3. 加速机制 三、游戏…