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

递归陷阱七例

目录

栈溢出

无限递归

大常数参数

递归深度过大

重复计算

函数调用开销

递归与迭代的选择

总结


递归是一种强大的编程技术,它允许函数调用自身。递归在很多情况下可以简化代码,使问题更容易理解和解决。然而,递归也容易导致一些常见的问题,这些问题被称为递归陷阱。本文将总结一些常见的递归陷阱,并提供示例代码来避免这些陷阱。

  • 栈溢出

递归函数会在每次调用自身时创建一个新的栈帧。如果递归深度过大,可能会导致栈溢出。为了避免栈溢出,我们可以限制递归深度,或者使用尾递归优化。

示例代码:计算斐波那契数列

#include <stdio.h>int fibonacci(int n) {if (n <= 1) {return n;}return fibonacci(n - 1) + fibonacci(n - 2);
}int main() {int n = 10;printf("Fibonacci %d = %d\n", n, fibonacci(n));return 0;
}

在上面的代码中,我们使用递归计算斐波那契数列。然而,这个递归函数的效率很低,因为它会重复计算很多子问题。为了避免栈溢出,我们可以使用动态规划或缓存技术来优化递归函数。

  • 无限递归

递归函数必须有终止条件,否则它会无限递归下去。在编写递归函数时,一定要确保有正确的终止条件。

示例代码:计算阶乘

#include <stdio.h>int factorial(int n) {if (n == 0) {return 1;}return n * factorial(n - 1);
}int main() {int n = 5;printf("Factorial %d = %d\n", n, factorial(n));return 0;
}

在上面的代码中,我们使用递归计算阶乘。这个递归函数有一个明确的终止条件:当n等于0时,返回1。这样,递归函数就可以正确地计算出阶乘。

  • 大常数参数

递归函数的参数应该尽量小,以减少栈空间的使用。如果递归函数的参数过大,可能会导致栈溢出。

示例代码:计算幂

#include <stdio.h>double power(double x, int n) {if (n == 0) {return 1;}return x * power(x, n - 1);
}int main() {double x = 2.0;int n = 10;printf("%.2f^%d = %.2f\n", x, n, power(x, n));return 0;
}

在上面的代码中,我们使用递归计算幂。然而,这个递归函数的参数n是一个整数,如果n非常大,可能会导致栈溢出。为了避免这个问题,我们可以使用迭代而不是递归。

  • 递归深度过大

有些问题本身就需要很深的递归深度才能解决。在这种情况下,我们可以尝试使用非递归算法,或者使用分治法将问题分解成更小的子问题。

示例代码:汉诺塔

#include <stdio.h>void hanoi(int n, char from, char to, char aux) {if (n == 1) {printf("Move disk 1 from %c to %c\n", from, to);return;}hanoi(n - 1, from, aux, to);printf("Move disk %d from %c to %c\n", n, from, to);hanoi(n - 1, aux, to, from);
}int main() {int n = 3;hanoi(n, 'A', 'C', 'B');return 0;
}

在上面的代码中,我们使用递归解决汉诺塔问题。这个问题需要很深的递归深度才能解决。为了避免栈溢出,我们可以限制递归深度,或者使用非递归算法。

  • 重复计算

在递归函数中,可能会重复计算相同的子问题多次。为了避免重复计算,我们可以使用记忆化递归(也称为递归+缓存)。

示例代码:计算斐波那契数列(记忆化递归)

#include <stdio.h>
#include <stdlib.h>int *fibCache;int fibonacci(int n) {if (n <= 1) {return n;}if (fibCache[n] != -1) {return fibCache[n];}fibCache[n] = fibonacci(n - 1) + fibonacci(n - 2);return fibCache[n];
}int main() {int n = 10;fibCache = (int *) calloc(n + 1, sizeof(int));for (int i = 0; i <= n; i++) {fibCache[i] = -1;}printf("Fibonacci %d = %d\n", n, fibonacci(n));free(fibCache);return 0;
}

在上面的代码中,我们使用记忆化递归计算斐波那契数列。我们创建了一个缓存数组fibCache来存储已经计算过的斐波那契数。在递归函数中,我们首先检查fibCache[n]是否已经计算过,如果已经计算过,就直接返回结果,否则计算fibCache[n],并将结果存储在fibCache[n]中。

  • 函数调用开销

递归函数的每次调用都会有一定的开销,包括参数传递、栈帧创建和销毁等。在递归深度较大时,这些开销可能会累积起来,影响程序的性能。为了避免这个问题,我们可以尝试减少递归深度,或者使用非递归算法。

示例代码:计算幂(迭代)

#include <stdio.h>double power(double x, int n) {double result = 1.0;while (n > 0) {if (n % 2 == 1) {result *= x;}x *= x;n /= 2;}return result;
}int main() {double x = 2.0;int n = 10;printf("%.2f^%d = %.2f\n", x, n, power(x, n));return 0;
}

在上面的代码中,我们使用迭代而不是递归计算幂。这个迭代算法的时间复杂度是O(log n),与递归算法相当,但它避免了递归调用的开销。

  • 递归与迭代的选择

在解决某些问题时,递归和迭代都是可行的选择。一般来说,递归更容易理解和实现,但可能会导致性能问题。而迭代可能更难理解和实现,但通常更高效。在选择递归还是迭代时,我们应该根据问题的性质和性能要求来决定。

示例代码:计算斐波那契数列(迭代)

#include <stdio.h>int fibonacci(int n) {int a = 0, b = 1, temp;while (n > 0) {temp = a + b;a = b;b = temp;n--;}return a;
}int main() {int n = 10;printf("Fibonacci %d = %d\n", n, fibonacci(n));return 0;
}

在上面的代码中,我们使用迭代计算斐波那契数列。这个迭代算法的时间复杂度是O(n),与递归算法相当,但它避免了递归调用的开销。

  • 总结

递归是一种强大的编程技术,但容易导致一些常见的问题。为了避免递归陷阱,我们应该限制递归深度,使用尾递归优化,确保有正确的终止条件,尽量使用小常数参数,或者使用非递归算法。在编写递归函数时,我们应该仔细考虑这些问题,并选择合适的方法来解决它们。

在本文中,我们讨论了一些常见的递归陷阱,并提供了相应的示例代码。通过理解和避免这些陷阱,我们可以更有效地使用递归来解决各种问题。

相关文章:

递归陷阱七例

目录 栈溢出 无限递归 大常数参数 递归深度过大 重复计算 函数调用开销 递归与迭代的选择 总结 递归是一种强大的编程技术&#xff0c;它允许函数调用自身。递归在很多情况下可以简化代码&#xff0c;使问题更容易理解和解决。然而&#xff0c;递归也容易导致一些常见的…...

【3D基础】坐标转换——地理坐标投影到平面

汤国安版GIS原理第二章重点 1.常见投影方式 https://download.csdn.net/blog/column/9283203/83387473 Web Mercator投影&#xff08;Web Mercator Projection&#xff09;&#xff1a; 优点&#xff1a; 在 Web 地图中广泛使用&#xff0c;易于显示并与在线地图服务集成。在…...

颈椎锻炼方式

1. 颈部伸展运动&#xff1a;坐直&#xff0c;慢慢将头向前伸展&#xff0c;直到感到轻微的拉伸&#xff0c;保持数秒钟&#xff0c;然后缓慢放松。重复10次。 2. 颈部旋转运动&#xff1a;坐直&#xff0c;慢慢将头向一侧转动&#xff0c;直到感到轻微的拉伸&#xff0c;保持…...

测试环境搭建:JDK+Tomcat+Mysql+Redis

基础的测试环境搭建&#xff1a; LAMPLinux(CentOS、ubuntu、redhat)ApacheMysqlPHP LTMJLinux(CentOS、ubuntu、redhat)TomcatMysql(Oracle)RedisJava 真实的测试环境搭建&#xff1a;&#xff08;企业真实的运维&#xff09; 基于SpringBoot&#xff08;SpringCloud分布式微…...

(delphi11最新学习资料) Object Pascal 学习笔记---第11章第1节(混合引用中的错误)

11.1.3 混合引用中的错误 ​ 在使用对象时&#xff0c;你通常应该只使用对象变量或接口变量来访问它们。混合使用这两种方法会破坏对象 Pascal 所提供的引用计数机制&#xff0c;并可能导致极难跟踪的内存错误。在实践中&#xff0c;如果你决定使用接口&#xff0c;你可能应该…...

代码随想录算法训练营第三天 | 链表理论基础,203.移除链表元素,707.设计链表,206.反转链表

对于链表完全陌生&#xff0c;但是看题目又觉得和数组一样的 链表理论基础 Q&#xff1a;什么是链表&#xff1f; A&#xff1a;链表是由一系列结点组成的。每一个结点由两部分组成&#xff1a;数据和指针。 203.移除链表元素 题目&#xff1a; 给你一个链表的头节点 head 和…...

如何利用仪表构造InfiniBand流量在数据中心测试中的应用

一、什么是Infiniband&#xff1f; 在当今数据爆炸的时代&#xff0c;数据中心作为信息处理的中心枢纽&#xff0c;面临着前所未有的挑战。传统的通信方式已经难以满足日益增长的数据传输需求&#xff0c;而InfiniBand技术的出现&#xff0c;为数据中心带来了全新的通信解决方…...

Kubernetes 文档 / 概念 / Kubernetes 架构 / 节点

Kubernetes 文档 / 概念 / Kubernetes 架构 / 节点 此文档从 Kubernetes 官网摘录 中文地址 英文地址 节点上的组件包括 kubelet、 容器运行时以及 kube-proxy。 管理 向 API 服务器添加节点的方式主要有两种&#xff1a; 节点上的 kubelet 向控制面执行自注册&#xff1b…...

ICode国际青少年编程竞赛- Python-1级训练场-for循环练习

ICode国际青少年编程竞赛- Python-1级训练场-for循环练习 1、 for i in range(3):Dev.step(4)Dev.turnLeft()2、 for i in range(3):Dev.step(2)Dev.turnRight()Dev.step(2)Dev.turnLeft()3、 for i in range(3):Dev.step(2)Dev.turnRight()Dev.step(2)Dev.turnLeft()4、 for…...

Flutter分模块开发、模块可单独启动、包含Provider

前言 当前案例 Flutter SDK版本&#xff1a;3.13.2 目前Flutter都是在一个项目中&#xff0c;创建不同目录进行模块开发&#xff0c;我进行Android原生开发时&#xff0c;发现原生端&#xff0c;是可以将每个模块独立运行起来的&#xff0c;灵感来自这&#xff1b; 折腾了几…...

Element-UI快速入门:构建优雅的Vue.js应用界面

Element-UI是一套基于Vue.js的组件库&#xff0c;提供了丰富的UI组件和交互效果&#xff0c;帮助开发者快速构建出美观、功能丰富的Web应用界面。本文将介绍如何快速入门Element-UI&#xff0c;并搭建一个简单的示例界面。 步骤一&#xff1a;安装Element-UI 首先&#xff0c…...

Flutter 中的 @immutable:深入解析与最佳实践

在 Flutter 开发中&#xff0c;immutable 注释扮演着至关重要的角色&#xff0c;用于标记不可变类。不可变类顾名思义&#xff0c;其状态一旦创建便不可更改&#xff0c;这与可变类截然不同。后者允许在创建后对实例进行修改。 immutable 的利好 引入不可变类可以带来诸多优势…...

Pandas数据可视化 - Matplotlib、Seaborn、Pandas Plot、Plotly

可视化工具介绍 让我们一起探讨Matplotlib、Seaborn、Pandas Plot和Plotly这四个数据可视化库的优缺点以及各自的适用场景。这有助于你根据不同的需求选择合适的工具。 1. Matplotlib 优点: 功能强大&#xff1a;几乎可以用于绘制任何静态、动画和交互式图表。高度可定制&a…...

人工智能的发展将如何重塑网络安全

微信搜索关注公众号网络研究观&#xff0c;获取更多信息。 人们很容易认为人工智能 (AI) 真正出现是在 2019 年&#xff0c;当时 OpenAI 推出了 ChatGPT 的前身 GPT-2。 但现实却有些不同。人工智能的基础可以追溯到 1950 年&#xff0c;当时数学家艾伦图灵发表了题为“计算机…...

Prometheus+Grafana多方位监控

PrometheusGrafana多方位监控 契机 ⚙ 最近发现火山引擎有托管的Prometheus,可是当前是邀测阶段。并且发现火山云的ECS是自带开机自启的exporter的。刚好需要搭建一套服务器监控&#xff0c;所以研究了一套Prometheus监控&#xff0c;包含linux主机监控nginx监控es监控rabbitM…...

使用Docker安装Redis

大家好&#xff0c;今天给大家分享一下如何使用docker安装Redis&#xff0c;关于docker的安装和常用命令&#xff0c;大家可以参考下面两篇文章&#xff0c;本文中不做过多描述。 Docker在Windows与CentOS上的安装 Docker常用命令 关于Redis的介绍与常用操作可以参考&#x…...

React 之 Effect与事件(event)(八)

Effect&#xff08;useEffect Hook&#xff09; 在React中&#xff0c;Effect&#xff08;或者更具体地说&#xff0c;useEffect Hook&#xff09;是一个特殊的函数&#xff0c;它允许你在函数组件中执行副作用操作。这些副作用操作可能包括数据获取、手动更改DOM、订阅或取消订…...

网卡的了解

什么是网卡_csdn网卡是什么-CSDN博客 MAC地址&#xff1a;48位串行号&#xff08;独一无二&#xff09; 2^48281 474 976 710 656 10位&#xff1a;10亿 5位&#xff1a;1万 15位&#xff1a;10万亿 网卡就是网络适配器 设置--->网络和Internet--->高级网络设置--->硬…...

SSM框架目录

ssm 知识相关目录主要参考尚硅谷 赵伟风老师的视屏&#xff0c;参考链接为 SSM视频_ SSM技术视频_SSM视频教程_尚硅谷 【注意】有些图片为了简便&#xff0c;所以就直接使用了视屏分析。 1、SSM框架相关知识 SpringFramework 基本概念 链接&#xff1a;SpringFramework 基本…...

MATLAB实现杜拉德公式和凯夫公式的计算固液混合料浆临界流速

MATLAB实现杜拉德公式和凯夫公式的计算固液混合料浆临界流速: 杜拉德公式是用来计算非均质固液混合料浆在输送管中的临界速度的公式&#xff0c;具体形式为&#xff1a; uL FL (2gD / (ρ0 - ρ1))^(1/2) 其中&#xff1a; uL&#xff1a;表示料浆的临界速度&#xff0c;…...

ClickHouse:开源数据引擎在AI浪潮爆发,挑战传统数据库巨头

ClickHouse&#xff1a;开源数据引擎爆发&#xff0c;在AI浪潮中挑战传统数据库巨头过去18个月&#xff0c;开源数据基础设施里最热的公司除了Supabase可能就是ClickHouse了。ClickHouse Cloud ARR在2025年保持250%的同比增速&#xff0c;第三方估计从2024年中的约1500万美元增…...

从手机解锁合法化看DMCA、消费者权利与设备所有权的博弈

1. 从“越狱”到合法化&#xff1a;一场关于设备所有权的消费者权利运动2013年初&#xff0c;如果你在美国买了一部合约机&#xff0c;然后想把它带到另一家运营商使用&#xff0c;你面临的不仅仅是不兼容的技术问题&#xff0c;还可能是一项重罪——最高五年的监禁和五十万美元…...

喷墨设备怎么选?2026年UV喷码技术深度评测与选购指南

面对市场上琳琅满目的工业喷墨设备&#xff0c;尤其是UV喷墨设备厂家&#xff0c;采购者如何做出明智选择&#xff1f;本文将从技术前沿、核心参数与行业应用三大维度&#xff0c;为您提供一份详尽的评测与选购指南&#xff0c;并深度剖析以中防uv喷码机为代表的专业制造商如何…...

PHP使用Intervention Image图像处理

在 Web 开发中&#xff0c;图像处理是常见需求&#xff1a;缩略图生成、水印添加、格式转换……但原生 PHP 的 GD 库 API 复杂且易出错。本文将提供Intervention ImagePHP 图像处理库从安装到高级用法的完整实战指南&#xff0c;它能用极简代码完成复杂的图像操作。 目录 安装…...

模块三-数据清洗与预处理——14. 重复值处理

14. 重复值处理 1. 概述 重复值是数据中的常见问题&#xff0c;可能来自数据录入错误、系统重复导出、数据合并等原因。重复数据会导致统计偏差、模型过拟合&#xff0c;需要在数据预处理阶段处理。 import pandas as pd import numpy as np# 创建包含重复值的示例数据 df pd.…...

MyScaleDB:基于SQL的向量数据库实战,实现混合查询与AI应用开发

1. 项目概述&#xff1a;当向量数据库遇见SQL如果你最近在折腾大模型应用&#xff0c;尤其是想给AI应用加上“长期记忆”或者实现精准的文档问答&#xff0c;那你大概率已经听过“向量数据库”这个词。从早期的Milvus、Pinecone&#xff0c;到后来各大云厂商纷纷入局&#xff0…...

Deep Multiview Clustering by Contrasting Cluster Assignments

通过对比不同的聚类分配实现深度多视图聚类摘要深度学习在大规模多视图聚类上表现好&#xff0c;但是该领域如何学习不同视图的潜在表示仍是一个问题。作者认为不同视图之间应该对齐的不是中间层特征&#xff0c;而是最后的聚类分配结果。因此提出 CVCL&#xff0c;通过对比不同…...

从古代数学到信息学奥赛:秦九韶算法如何帮你秒杀多项式计算题?

从古代数学到信息学奥赛&#xff1a;秦九韶算法如何帮你秒杀多项式计算题&#xff1f; 在杭州西湖畔的岳王庙旁&#xff0c;矗立着一块刻有"大衍求一术"的石碑&#xff0c;这是南宋数学家秦九韶留给后人的智慧结晶。当我们今天面对一道看似普通的多项式计算题时&…...

Nodejs服务端应用接入Taotoken多模型API指南

&#x1f680; 告别海外账号与网络限制&#xff01;稳定直连全球优质大模型&#xff0c;限时半价接入中。 &#x1f449; 点击领取海量免费额度 Nodejs服务端应用接入Taotoken多模型API指南 对于Node.js后端开发者而言&#xff0c;将大模型能力集成到Web服务或API中&#xff0…...

为AI编程助手构建本地知识库:YAP项目实战指南

1. 项目概述&#xff1a;当AI编程助手遇上专属知识库如果你和我一样&#xff0c;日常重度依赖Cursor这类AI编程助手&#xff0c;那你一定遇到过这样的场景&#xff1a;面对一个复杂的内部项目&#xff0c;或者一个使用了大量私有库、自定义框架的代码库&#xff0c;Cursor的响应…...