当前位置: 首页 > article >正文

深度剖析 C 语言函数递归:原理、应用与优化

    在 C 语言的函数世界里,递归是一个独特且强大的概念。它不仅仅是函数调用自身这么简单,背后还蕴含着丰富的思想和广泛的应用。今天,让我们跟随这份课件,深入探索函数递归的奥秘。

一、递归基础:概念与思想

    递归是一种解决问题的方法,在 C 语言中表现为函数自己调用自己。这一概念乍看有些像循环,但又有着本质的区别。以一个简单的示例来说明:

    这段代码展示了递归的基本形式,但它是一个会导致栈溢出的死递归,因为没有设置终止条件。递归的核心思想是把一个大型复杂问题层层转化为与原问题相似但规模较小的子问题来求解,直到子问题小到可以直接解决,这个过程就像是把大象放进冰箱,分步骤逐步拆解,最终达成目标,也就是所谓的 “大事化小”

二、递归的关键:限制条件

    递归要正常工作,必须满足两个关键限制条件:

    存在终止条件:当满足特定条件时,递归不再继续。这就好比在一条路上设置了终点线,到达终点就停止前进。在代码中,通常表现为一个判断语句,例如在计算阶乘的递归函数中,if(n==0)就是终止条件。

    逐步接近终止条件:每次递归调用后,问题规模应逐渐减小,更接近终止条件。就像跑步比赛,每跑一步都离终点更近一点。在递归函数中,每次调用都要让参数朝着满足终止条件的方向变化,如计算阶乘时,n每次递归都减 1。

三、递归的经典应用

(一)计算 n 的阶乘

    计算 n 的阶乘是递归的经典案例。阶乘的定义是所有小于及等于该数的正整数的积,0 的阶乘为 1,数学公式为n!= n ∗(n −1)!。基于此,我们可以编写如下递归函数:

    这个函数中,当n为 0 时,函数直接返回 1,这是递归的终止点。当n大于 0 时,函数通过不断调用自身,将n逐渐减小,最终完成阶乘的计算。例如计算 5 的阶乘,函数会依次计算5 * 4!4 * 3!3 * 2!2 * 1!,直到 1 的阶乘(即 1),然后逐步返回计算结果。

    运⾏结果(这⾥不考虑n太⼤的情况,n太⼤存在溢出);

(二)顺序打印整数的每一位

    输入一个整数,按顺序打印其每一位,例如输入 1234,输出 1 2 3 4。解决这个问题的关键在于如何拆分整数的每一位。通过%10操作可以得到整数的最低位,再通过/10操作去掉已处理的最低位。利用递归思想,可以将打印一个多位数的问题转化为打印去掉最低位后的数,再打印最低位。代码实现如下:

    在这个函数中,当n是一位数时,直接打印n,这是递归的终止条件。当n超过一位数时,先递归调用Print(n/10)打印除最低位外的其他位,然后再打印最低位n%10

四、递归与迭代的对比

    递归虽然强大,但并非没有缺点。在递归函数调用过程中,每次调用都需要在内存栈区申请一块空间来保存局部变量和参数,这块空间称为函数栈帧。如果递归层次过深,会占用大量栈帧空间,可能导致栈溢出问题。例如在计算斐波那契数时,传统递归方式会出现效率低下的情况。

    相比之下,迭代(通常指循环)在一些场景下更具优势。以计算阶乘为例,迭代实现如下:

     这段代码通过循环从 1 累乘到n,避免了递归调用带来的额外开销,执行效率更高。在实际编程中,我们需要根据具体问题的特点,权衡递归和迭代的优缺点,选择最合适的方法

五、递归的拓展应用

(一)斐波那契数的计算

    斐波那契数列的特点是前两个数为 1,从第三个数开始,每个数都等于前两个数之和。用递归方式计算第 n 个斐波那契数的代码如下:

    然而,当n较大时,如n = 50,使用这种递归方式计算会花费极长的时间,因为递归过程中存在大量重复计算。为了优化,可以采用迭代方式:

迭代方式从前往后依次计算斐波那契数,避免了重复计算,大大提高了效率。

(二)青蛙跳台阶问题

    一只青蛙一次可以跳上 1 级台阶,也可以跳上 2 级台阶,求青蛙跳上 n 级台阶总共有多少种跳法。这是一个可以用递归很好解决的问题。假设跳上n级台阶的跳法数为F(n),则有F(n) = F(n-1) + F(n-2),这与斐波那契数列的递归公式相似。当n为 1 时,只有 1 种跳法;当n为 2 时,有 2 种跳法(一次跳 2 级或分两次每次跳 1 级)。递归实现代码如下:

    同样,为了提高效率,也可以将其转换为迭代实现。

(三)汉诺塔问题

汉诺塔问题是一个古老的益智游戏,有三根柱子 A、B、C,A 柱上有若干个盘子,盘子大小不等,大的在下,小的在上。要求将 A 柱上的盘子借助 B 柱全部移到 C 柱上,每次只能移动一个盘子,且在移动过程中,大盘子不能放在小盘子上面。这个问题可以用递归完美解决。假设要将n个盘子从 A 柱借助 B 柱移到 C 柱,递归思路如下:

  • n - 1个盘子从 A 柱借助 C 柱移到 B 柱。
  • 将第n个盘子从 A 柱移到 C 柱。
  • n - 1个盘子从 B 柱借助 A 柱移到 C 柱。

    递归在这些拓展问题中展现了强大的解题能力,通过巧妙地利用递归的 “大事化小” 思想,将复杂问题简化为可解决的子问题。但在实际应用中,也要注意递归可能带来的性能问题,根据情况选择合适的优化策略或实现方式。 

相关文章:

深度剖析 C 语言函数递归:原理、应用与优化

在 C 语言的函数世界里,递归是一个独特且强大的概念。它不仅仅是函数调用自身这么简单,背后还蕴含着丰富的思想和广泛的应用。今天,让我们跟随这份课件,深入探索函数递归的奥秘。 一、递归基础:概念与思想 递归是一种…...

goredis常见基础命令

基本操作 //删除键 exists,err: rdb.Exists(ctx,"key").Result() if err!nil{panic(err) } if exists>0{err rdb.Del(ctx,"key").Err()if err!nil{panic(err)} }string类型 //设置一个键值对 //0表示没有过期时间 err:rdb.Set(ctx,"key1",…...

【Linux网络】序列化、守护进程、应用层协议HTTP、Cookie和Session

⭐️个人主页:小羊 ⭐️所属专栏:Linux 很荣幸您能阅读我的文章,诚请评论指点,欢迎欢迎 ~ 目录 1、序列化和反序列化2、守护进程2.1 什么是进程组?2.2 什么是会话? 3、应用层协议HTTP3.1 HTTP协议3.2 HT…...

JavaScript函数-arguments的使用

在JavaScript编程语言中,函数是构建复杂逻辑和实现代码复用的关键组件。虽然现代JavaScript(尤其是ES6及之后版本)提供了更多灵活的方式来处理函数参数(如剩余参数、默认参数等),但arguments对象仍然是一个…...

Hadoop常用操作命令

在NameNode节点格式化集群 初始化集群 hdfs namenode -format启动HDFS sbin/start-dfs.sh启动yarn sbin/start-yarn.sh启动NodeManager yarn-daemon.sh start nodemanager启动DataNode hadoop-daemon.sh start datanode启动SecondaryNameNode hadoop-daemon.sh start se…...

system verilog的流操作符

流操作符&#xff0c;有分为操作对象是一整个数组和单独的数据两种&#xff0c;例如bit [7:0] a[4]和bit [31:0] b&#xff0c;前者操作对象是数组&#xff0c;后者是单独一个较大位宽的数。 流操作符有<<和>>&#xff0c;代表从右向左打包和从左向右打包。 打包的…...

LLM2CLIP论文学习笔记:强大的语言模型解锁更丰富的视觉表征

1. 写在前面 今天分享的一篇论文《LLM2CLIP: P OWERFUL L ANGUAGE M ODEL U NLOCKS R ICHER V ISUAL R EPRESENTATION》&#xff0c; 2024年9月微软和同济大学的一篇paper&#xff0c; 是多模态领域的一篇工作&#xff0c;主要探索了如何将大模型融合到Clip模型里面来进一步提…...

计算机毕业设计SpringBoot+Vue.jst网上超市系统(源码+LW文档+PPT+讲解)

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…...

python-静态方法和类方法

Java之类的编程语言还带有静态方法&#xff0c;Python类也拥有与静态方法明确对应的方法。此外&#xff0c;Python还拥有类方法&#xff0c;要比静态方法更高级一些。 静态方法与Java一样&#xff0c;即便没有创建类的实例&#xff0c;静态方法也是可以调用的&#xff0c;当然…...

什么是手机9008模式?如何进入9008

之前给大家分享了一些有关手机刷机的知识&#xff0c;今天给大家讲一讲如果刷机过程中不慎变砖应该如何应对&#xff08;当然了&#xff0c;希望大家都不会遇到&#xff09;&#x1f602;&#x1f604; 在给手机 Root 或刷机时&#xff0c;线刷 9008 指的是利用 高通 9008 模式…...

嵌入式之指针

在嵌入式系统中指针是一种非常重要的概念。它们用于直接访问内存地址&#xff0c;能够提高程序的灵活性和效率。 一、基本概念 1. 指针的基本概念 定义&#xff1a;指针是一个变量&#xff0c;其值为另一个变量的地址。通过指针&#xff0c;可以间接访问和修改该变量的值。声…...

网络安全研究

1.1 网络安全面临的威胁 网络安全面临的威胁呈现出多样化和复杂化的趋势&#xff0c;给个人、企业和国家的安全带来了严峻挑战。以下是当前网络安全面临的主要威胁&#xff1a; 1.1.1 数据泄露风险 数据泄露是当前网络安全的重大威胁之一。根据国家互联网应急中心发布的《20…...

Git入门:数据模型 to 底层原理

版本控制系统&#xff08;VCS&#xff09;是软件开发中不可或缺的工具&#xff0c;而Git作为现代版本控制的事实标准&#xff0c;其底层设计远比表面命令更加优雅。本文将从数据模型的角度&#xff0c;揭示Git的核心工作原理。 Git的核心概念 1. 快照&#xff08;Snapshot&am…...

openharmony中hdf框架的驱动消息机制的实现原理

openharmony中hdf框架的驱动消息机制的实现原理 在分析hdf框架时发现绕来绕去的&#xff0c;整体梳理画了一遍流程图&#xff0c;发现还是有点模糊甚至不清楚如何使用的&#xff0c;详细的每个点都去剖析细节又过于消耗时间&#xff0c;所以有时间便从功能应用的角度一块块的去…...

HTTP SSE 实现

参考&#xff1a; SSE协议 SSE技术详解&#xff1a;使用 HTTP 做服务端数据推送应用的技术 一句概扩 SSE可理解为&#xff1a;服务端和客户端建立连接之后双方均保持连接&#xff0c;但仅支持服务端向客户端推送数据。推送完毕之后关闭连接&#xff0c;无状态行。 下面是基于…...

二分图检测算法以及最大匹配算法(C++)

上一节我们学习了有向图中的最大连通分量. 本节我们来学习二分图. 二分图是一种特殊的图结构, 能够帮助我们高效地解决这些匹配和分配问题. 本文将带你了解二分图的基本概念, 判定方法, 最大匹配算法以及实际应用场景. 环境要求 本文所用样例在Windows 11以及Ubuntu 24.04上面…...

Keepalive基础

一。简介和功能 vrrp协议的软件实现&#xff0c;原生设计目的是为了高可用ipvs服务 功能&#xff1a; 1.基于vrrp协议完成地址流动 2.为vip地址所在的节点生成ipvs规则&#xff08;在配置文件中预先定义&#xff09; 3.为ipvs集群的各RS做健康状况检测 4.基于脚本调用接口…...

计算机毕业设计SpringBoot+Vue.jst0图书馆管理系统(源码+LW文档+PPT+讲解)

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…...

【Java消息队列】应对消息丢失、重复、顺序与积压的全面策略

应对消息丢失、重复、顺序与积压的全面策略 引言kafka消息丢失生产者消费者重复消费顺序消费消息积压生产者消费者其他RabbitMQ消息丢失生产者事务机制,保证生产者发送消息到 RabbitMQ Server发送方确认机制,保证消息能从交换机路由到指定队列保证消息在 RabbitMQ Server 中的…...

AI大模型学习(三): LangChain(二)

Langchain构建聊天机器人 安装依赖 pip install langchain_community Chat History:它允许聊天机器人"记住"过去的互动,并在回应后续问题时考虑他们 代码 # 创建模型 from langchain_core.messages import HumanMessage from langchain_core.prompts import ChatP…...

apply的用法

apply 是一个在编程语言中常见的函数&#xff0c;它在不同的上下文和语言中有不同的用途。以下是 apply 在常见编程语言中的几种常见用法&#xff1a; 1. Python 中的 apply 方法 在 Python 中&#xff0c;apply 主要用于 pandas 库中的 DataFrame 或 Series 对象&#xff0c…...

【论文解读】TransMLA: Multi-Head Latent Attention Is All You Need

论文链接 1. 论文背景与问题动机 现代大规模语言模型&#xff08;LLM&#xff09;在推理时往往遇到通信瓶颈&#xff0c;主要原因在于自注意力机制中需要缓存大量的 Key-Value&#xff08;KV&#xff09;对。例如&#xff0c;对于 LLaMA‑65B 这种模型&#xff0c;即使采用 8…...

CentOS 下安装和配置 HTTPD 服务的详细指南

CentOS 下安装和配置 HTTPD 服务的详细指南 CentOS 下安装和配置 HTTPD 服务的详细指南1. 环境准备2. 安装 HTTPD 服务2.1 更新系统2.2 安装 HTTPD2.3 启动 HTTPD 服务2.4 检查 HTTPD 服务状态 3. 配置防火墙3.1 开放 HTTP 和 HTTPS 端口3.2 验证防火墙规则 4. 配置 HTTPD4.1 主…...

VUE3中子组件改变父组件传过来的值(props)的方法和使用场景详解

在 Vue 3 中&#xff0c;子组件改变父组件传过来的值&#xff08;props&#xff09;的方法主要有以下几种&#xff1a;通过事件发射、使用 v-model、模拟 .sync 修饰符的功能&#xff08;Vue 3 中已移除&#xff09;&#xff0c;以及使用 ref 或 reactive。下面我将结合代码示例…...

登录-06.JWT令牌-生成和校验

一.JWT令牌的生成和校验 JWT令牌生成 想要生成JWT令牌&#xff0c;那么就要首先引入JWT令牌的相关依赖&#xff0c; <dependency><groupId>io.jsonwebtoken</groupId><artifactId>jjwt-api</artifactId><version>0.11.2</version>…...

【Git】多人协作

文章目录 完成准备工作多人协作场景一场景二远程分支删除后&#xff0c;本地 git branch -a 依然能看到的解决办法 完成准备工作 在之前&#xff0c;我们所完成的工作如下&#xff1a; 基本完成 Git 的所有本地库的相关操作&#xff0c;git基本操作&#xff0c;分支理解&#…...

Python爬虫-破解字体加密技术

前言 本文是该专栏的第77篇,后面会持续分享python爬虫干货知识,记得关注。 字体加密是一种常见的反爬虫技术,通过自定义字体文件和字符映射来保护网页内容,防止爬虫直接获取文本信息。 在文章《Python爬虫-猫眼电影的影院数据》中,笔者有详细介绍过猫眼的相关数据采集。…...

邮件安全之发件人伪造

电子邮件工作原理 电子邮件传输过程中主要涉及到SMTP、IMAP、POP3三种协议&#xff0c;具体功能如下&#xff1a; SMTP:全称Simple Mail Transfer Protocol&#xff0c;即简单邮件传输协议&#xff0c;主要用于发送邮件&#xff0c;使用端口号25。 IMAP:全称Internet Mail Acce…...

git 常用功能

以下是 Git 的常用功能及其命令&#xff1a; 初始化仓库 git init在当前目录初始化一个新的 Git 仓库。 克隆仓库 git clone <仓库地址>将远程仓库克隆到本地。 查看状态 git status查看工作区和暂存区的状态。 添加文件到暂存区 git add <文件名>将文件添…...

【llm落地】从零到一,用DeepSeek打造智能BI工具:自然语言驱动数据洞察

在数据驱动的时代,商业智能 (BI) 工具已经成为企业决策的关键。然而,传统的 BI 工具往往操作复杂,需要专业技能才能驾驭。想象一下,如果用户只需要用 自然语言 就能轻松查询数据、获取分析结果甚至生成可视化图表,那将会多么高效和便捷! 本文将带你踏上从零到一构建智能…...