【Hello Algorithm】认识一些简单的递归

news2024/11/19 13:27:02

本篇博客介绍: 认识一些简单的递归

认识一些简单的递归

    • 打印一个字符串全部的子序列
    • 打印一个字符串的全排列
    • 不申请额外的空间 逆序输出一个栈

我在刚刚学习C语言的时候写过一个汉诺塔问题 大家可以参考下我之前写的这篇博客

汉诺塔问题

其实这个问题也可以这么解决

  • 我们设计六个函数 这六个函数分别代表从a到b 从a到c …
  • 但是实际上我们使用了一个函数就解决了上面六个函数才能解决的事

这就说明 我们可以使用增加参数的方式来增加递归函数的可能性

打印一个字符串全部的子序列

题目要求如下

假设字符串String str = “abc”,那么它的子序列有" "、a、b、c、ab、ac、bc、abc。依次从头往后拿,所有情况都枚举即可

我们回顾下题目实际上就是从第一个字符串开始我们考虑每个字符要和不要两种情况

那么我们可以设计这样子的一个递归函数

void process(string str ,int index , list<string> ans , string path)

这四个参数分别代表着

  • 我们要传入的字符串
  • 现在走到的位置
  • 答案
  • 目前为止拼接的字符串

我们每个递归函数内部只需要考虑当前字符是否拼接 之后传递给下一个递归函数即可

完整代码如下

  
  void process(const string& str ,int index , list<string>& ans , string path1)
  {
    if (str.length() == index)
    {
      ans.push_back(path1);
      return;    
    }    
      
    process(str , index + 1 , ans , path1);    
    process(str , index + 1 , ans , path1 + str[index]);    
  }

如果不允许有重复的字符串出现应该怎么做呢?

此时我们就可以使用 unordered_set 容器来自动去重即可

打印一个字符串的全排列

输入一个字符串 打印这个字符串的全排列

比如说给你一个字符串 abc

实际上就是给你一个字符集合 {a, b, c} 要求你将这三个字符随意组合出一个新的字符 共有多少种排列方式

实际上就是 3 x 2 x 1 一共六种

我们解决这个问题使用递归来解决 首先递归函数的声明如下

void  process(vector<char> lc , list<string>& ans , string path1)

代码如下

  void  process(vector<char> lc , list<string>& ans , string path1)    
  {    
    if (lc.size() == 0)    
    {    
      ans.push_back(path1);    
      return;    
    }    
    else    
    {    
      for (int i = 0; i < lc.size(); i++)    
      {    
        char cur = lc[i];    
        lc.erase(lc.begin() + i);    
        process(lc , ans , path1 + cur);                                                                                        
        lc.insert(lc.begin() + i , cur);    
      }    
    }    
  }  

解释下上面这段代码 我们每次都从集合中拿走一个元素 如果说集合被拿空了 那么此时的字符串肯定已经拼接完成了

如果集合还没空我们就一个个将这些字符串拼接起来 (注意! 如果你是将集合拼接在path中则要在最后还原path 如果是将path和cur一起传递则不用)

最后我们就能得到一个字符串的全排列了

在这里插入图片描述

在这里插入图片描述

当然 我们上面写的其实是一种不太好的递归 因为它可变参数的设计不太行 这样子我们递归到动态规划就会很麻烦 优秀的递归设计应该是下面这样子的

  void process(vector<char>& nums , int index , vector<string>& ans)    
  {    
    if (nums.size() == index)    
    {    
      string ans1;    
      for (auto x : nums)    
      {    
        ans1 += x;    
      }    
      ans.push_back(ans1);    
      return;    
    }    
    else    
    {    
      for (size_t i = index ; i < nums.size() ; i++)    
      {    
        swap(nums[i] , nums[index]);    
        process(nums , index + 1 , ans);    
        swap(nums[i] , nums[index]);    
      }                                                                                                                         
    }    
  } 

给定我们一个原数组 我们从数组中选择一个字符作一号位

for (size_t i = index ; i < nums.size() ; i++)
swap(nums[i] , nums[index]);  

选择完一号位之后我们再选二号位 之后依次递归即可

process(nums , index + 1 , ans);  

如果要求字符串不重复应该怎么做呢?

如果要求字符串不重复 这里有两种方式

  1. 我们使用unordered_set容器来保存答案
  2. 进行剪枝

我们可以设置一个bool数组 如果说某个元素已经做个头了 我们就不让它做头了

未剪枝前 在这里插入图片描述

剪枝代码
在这里插入图片描述
剪枝后

在这里插入图片描述

不申请额外的空间 逆序输出一个栈

我们现在申请一个栈 现在它内部有 1 2 3 4 5 6

如果我们正常输出 应该是这样子的

在这里插入图片描述

现在我们要逆序这个栈的输出 我们可以这么做

void revprint(stack<int>& st)    
{    
  if (st.empty())    
  {    
    return;    
  }    
  else    
  {    
    int top = st.top();    
    st.pop();    
    revprint(st);    
    cout << top << " ";                                                                                                         
  }    
}  

在这里插入图片描述
使用系统栈 等消栈的时候打印即可

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

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

相关文章

【RK3588】YOLO V5在瑞芯微板子上部署问题记录汇总

YOLO V5训练模型部署到瑞芯微的板子上面&#xff0c;官方是有给出案例和转过详情的。并且也提供了Python版本的推理代码&#xff0c;以及C语言的代码。 但是&#xff0c;对于转换过程中的细节&#xff0c;哪些需要改&#xff1f;怎么改&#xff1f;如何改&#xff0c;和为什么…

Altium Designer20.2.3安装详解

Altium Designer20.2.3是一个画PCB电路板的软件&#xff0c;今天有时间安装一下&#xff0c;接下来的一段时间就学习这个电路板的绘制了。特此记录下安装过程。 首先是下载软件&#xff0c;我直接放到我的网盘里面了。我把他分享出来吧&#xff01;希望能帮到更多的小伙伴&…

微信小程序开发缺少中间证书问题(腾讯云、阿里云等做服务器)

项目使用nginx做负载均衡后&#xff0c;不再采用原来直接用jar包的方式直接开启对应端口&#xff0c;所以需要重新从云服务器上下载证书&#xff0c;写入到Nginx读取的证书路径上即可。

SRTP协议与加密原理

1 SRTP简介 SRTP&#xff08;Secure Real-time Transport Protocol&#xff09;是一种用于保护实时通信数据的网络协议。它主要用于音频和视频通信&#xff0c;以确保数据的机密性和完整性。SRTP是在RTP&#xff08;Real-time Transport Protocol&#xff09;的基础上开发的&a…

基于安卓android微信小程序的旅游app系统

项目介绍 随着人民生活水平的提高,旅游业已经越来越大众化,而旅游业的核心是信息,不论是对旅游管理部门、对旅游企业,或是对旅游者而言,有效的获取旅游信息,都显得特别重要.自助定制游将使旅游相关信息管理工作规范化、信息化、程序化,提供旅游景点、旅游线路,旅游新闻等服务本…

【Verilog】采用采用模块结构建模,用1位全加器实现4位全加器详细步骤

题目要求&#xff1a; 采用模块结构建模&#xff0c;实例化四个1位全加器并连线&#xff0c;完成图示的四位全加器建模并编写四位全加器测试模块&#xff0c;在modelsim里执行&#xff0c;查看波形图。 首先&#xff0c;在工程区右键选择创建一个新文件去实现1位全加器的功能。…

图解AVL树的旋转操作

目录 AVL树的概念 AVL树结点的定义 AVL的插入 AVL树的旋转 左单旋 右单旋 左右双旋 右左双旋 AVL树的查找 AVL树的概念 二叉搜索树的缺点&#xff1a; 当构建二叉搜索树的数据有序或接近有序时二叉搜索树会退化为单链表。例如&#xff0c;当插入数据1&#xff0c;2&a…

Image resize, resample,rescale傻傻搞不懂看这里,大白话

Image resize, resample&#xff0c;rescale傻傻搞不懂 大白话 大家好&#xff0c;在学习图像处理的时候我们可能都对rescale&#xff0c;resize&#xff0c;resample感到困惑&#xff0c;在这里让我们进行学习一下。首先让我们现了解一下他们的大白话。resize就是改变大小&am…

项目部署之Jenkins

1. Jenkins介绍 Jenkins 是一款流行的开源持续集成&#xff08;Continuous Integration&#xff09;工具&#xff0c;广泛用于项目开发&#xff0c;具有自动化构建、测试和部署等功能。官网&#xff1a; http://jenkins-ci.org/。 Jenkins的特征&#xff1a; 开源的 Java语言…

ELF 文件介绍

前言 如果需要了解 动态加载,需要先了解需要动态加载的目标(对象):ELF 文件 ELF 文件用于 Linux、Unix等系统平台,与 Windows 上的 exe (PE)或者 DLL 文件格式不一样 ELF 简介 全称:Executable and Linking Format 中文:可执行和链接格式 ELF 文件格式是 Linux/Unix 通…

第一次微生物学实验

第一次微生物学实验 文章目录 前言一、显微镜油镜的使用以及细菌的简单染色法1.1. 实验目的1.2. 实验原理1.3. 实验材料1.4.实验步骤1.5. 实验结果及讨论1.5.1 实验结果1.5.2. 讨论 二、细菌的革兰氏染色以及芽孢染色法2.1. 实验目的2.2. 实验原理2.2.1. 革兰氏染色原理2.2.2. …

进程互斥的软件实现方法,硬件实现方法以及互斥锁

1.进程互斥的软件实现方法 1.单标志法 1.算法思想: 两个进程在访问完临界区后会把使用临界区的权限转交给另一个进程。 也就是说每个进程进入临界区的权限只能被另一个进程赋予。 2.例子 因此&#xff0c;该算法可以实现“同一时刻最多只允许一个进程访问临界区”。 3.主要…

网络初识必知会

局域网&#xff1a;把一些设备通过交换机/路由器连接起来 广域网&#xff1a;把更多的局域网也相互连接&#xff0c;当网络规模足够大的 交换机&#xff1a;组网过程中的重要设备&#xff01; 路由器&#xff1a;组网过程中的重要设备&#xff01; IP地址&#xff1a;描述一…

网络相关的基础知识整理

一、历史 1.1 早期阿帕网特点⭐⭐⭐ 没有纠错功能不能互联不同类型的计算机和不同类型的操作系统 1. 2 TCP/IP协议 点击【此处】跳转&#x1f517; TCP&#xff1a;用来检测网络传输中差错的传输控制协议IP&#xff1a;专门负责对不同网络进行互联的互联网协议&#xff08…

枚举,进制转换,char*,补码,算法,链表,位运算,NULL的含义

目录 什么是枚举 进制转换 字符指针 补码 算法 链表 算法 位运算符 ​编辑NULL的含义 什么是枚举 进制转换 4个二进制位组合才能有16个状态 字符指针 补码 编码就是解决这个东西到底用哪个二进制表示 不够位数指的是比如32位&#xff0c;前面都补1 #include <iostre…

Linux网络编程5-epoll模型

Linux网络编程5-epoll模型 1.epoll相关函数2.epoll服务器流程3.epoll服务器代码实现4.LT模式与ET模式5.ET模式一次性读完数据的实现6.epoll反应堆思想7.epoll反应堆代码实现8.epoll反应堆代码分析 1.epoll相关函数 #include <sys/epoll.h>int epoll_create(int size); /…

Bentley STAAD.Pro 2023 中文版 安装交流怎么安装?含解答

产品介绍 STAAD.Pro 是一个集结构建模、结构分析和结构设计于 一体的通用结构有限元程序。不仅具有常见的图形表格窗口 界面&#xff0c;而且有一个保存和修改建模分析设计命令的命令 编辑器。 另外&#xff0c;内置了几何建模向导和异形截面向导两个工具&#xff0c;为建模带来…

Gitlab+Jenkins自动化部署,解放双手

项目打包 ​ 在部署项目前需要对源码进行打包&#xff0c;一个简单的SpringBoot项目默认是打包为jar包&#xff0c;也就是在pom.xml中的<packaging>jar</packaging>方式&#xff0c;当然也会有一些打包成war包方式&#xff0c;使用外置的Tomcat应用服务器部署war包…

凉鞋的 Godot 笔记 106. 第二轮循环2D 场景视图Label

从这一篇开始&#xff0c;我们开始进行第二轮循环。 这次我们至少能够在游戏运行窗口能看到一些东西。 首先还是在场景窗口进行编辑&#xff0c;先创建一个节点: 在弹出的窗口&#xff0c;我们找到 Control/Label &#xff0c;如下所示: 点击创建&#xff0c;然后我们在 2D 的…

【dp】背包问题

背包问题 一、背包问题概述二、01背包问题&#xff08;1&#xff09;求这个背包至多能装多大价值的物品&#xff1f;&#xff08;2&#xff09;若背包恰好装满&#xff0c;求至多能装多大价值的物品&#xff1f; 三、完全背包问题&#xff08;1&#xff09;求这个背包至多能装多…