函数递归调用太深爆栈探索

2024-02-07 23:18

本文主要是介绍函数递归调用太深爆栈探索,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、常见的递归调用有斐波那契数列,1,1,2,3,5,8,13…
递归的缺点是递归的层数不能太多,会爆栈,下面从底层的角度讲述递归调用的原理。
1、链接逻辑
每个程序都有头文件,这些头文件包含一系列的库函数,当我们链接程序时这些目标模块就会装配成一个完整的模块装入。
每个模块都有自己的逻辑地址,链接完成后形成自己的从0开始的逻辑地址如图如图,各函数模块链接地址
2、装载
怎么将程序装入内存?程序一开始在磁盘上,还需要装入主存即内存才能运行。内存中低地址留给系统使用,高地址即用户区,是将原地址直接装入内存,相对地址是不变的
插入内存
计算机如何确定程序执行的下一条指令?是通过计算机中一个叫程序计数器(program Counter)的部件来存放程序要执行的下一条指令的。Cpu在程序计数器中取到下一条指令后就到内存中取这条指令,取到之后到Cpu中解析执行。如此循环往复,程序得以执行。
3、函数调用
函数调用,函数原本在主线上执行,遇到调用指令暂时到别的模块执行,执行结束后再回到主模块的过程
为什么可以跳到别的模块执行?因为函数的调用指令指明了模块的首地址,执行过程先将指令传给程序计数器(PC),此时中断了程序的顺序执行,指令之后的其他指令都被入在了内存中的一块专门实现函数调用的栈区,当程序调用指令执行到返回指令时,再出栈,将栈顶元素传给程序计数器(PC),程序因此又回到了主模块。
为什么要入栈?函数调用时入栈保护的不仅仅是指令地址,还有一些变量数据和局部变量,否则会丢失
4、递归函数中的调用
区别就是递归函数的调用就是自己调用自己。

int fibonacci(int n)
{
if(n==1||n==2) //假设本条指令的地址为10000return 1; //假设本条指令的地址为10001
int a=fibonacci(n-2); //假设本条指令的地址为10002
int b=fibonacci(n-1); //假设本条指令的地址为10003
int c=a+b; //假设本条指令的地址为10004
return c; //假设本条指令的地址为10005
}

假设主函数指令地址为15000,下一个指令是15001,从主函数到斐波那契数列的计算第五项,公式及示意图为:

f(5)=f(3)+f(4)=[f(1)+f(2)]+[f(2)+f(3)]=[f(1)+f(2)]+[f(2)+[f(1)+f(2)]]

在这里插入图片描述
循环往复,一直到指令15001出栈给程序计数器,回到主函数。计算完成
5、为什么爆栈?
内存中特地用来存放函数调用的栈区内存是有限的,如果递归太深,就会出现只入栈不出栈的情况。应谨慎使用
6、实际使用中像斐波那契的解法,使用递归会出现:1产生大量重复计算,2n过大时容易爆栈。可以使用动态规划解法(DP)

int fibonacci(int n)
{f[1]=1,f[2]=1;for(int i=3;i<=n;i++){f[i]=f[i-2]+f[i-1];}return f[n];
}

为什么说有重复计算,因为:
f3 = f1+f2
f5 = f3+f4
f4 = f2+f3
so f5=(f1+f2)+(f2+f3)
有两个f2,就相当于要计算两次f2
不如开一个比较大的数组,用循环来做。
(纯属笔记学习,抄自文章https://mp.weixin.qq.com/s/Re_0uZnJ9SikKoPG_bdSWQ)

这篇关于函数递归调用太深爆栈探索的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/689215

相关文章

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

MySQL count()聚合函数详解

《MySQLcount()聚合函数详解》MySQL中的COUNT()函数,它是SQL中最常用的聚合函数之一,用于计算表中符合特定条件的行数,本文给大家介绍MySQLcount()聚合函数,感兴趣的朋... 目录核心功能语法形式重要特性与行为如何选择使用哪种形式?总结深入剖析一下 mysql 中的 COUNT

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

MySQL 中 ROW_NUMBER() 函数最佳实践

《MySQL中ROW_NUMBER()函数最佳实践》MySQL中ROW_NUMBER()函数,作为窗口函数为每行分配唯一连续序号,区别于RANK()和DENSE_RANK(),特别适合分页、去重... 目录mysql 中 ROW_NUMBER() 函数详解一、基础语法二、核心特点三、典型应用场景1. 数据分

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN

Python中Tensorflow无法调用GPU问题的解决方法

《Python中Tensorflow无法调用GPU问题的解决方法》文章详解如何解决TensorFlow在Windows无法识别GPU的问题,需降级至2.10版本,安装匹配CUDA11.2和cuDNN... 当用以下代码查看GPU数量时,gpuspython返回的是一个空列表,说明tensorflow没有找到

Python get()函数用法案例详解

《Pythonget()函数用法案例详解》在Python中,get()是字典(dict)类型的内置方法,用于安全地获取字典中指定键对应的值,它的核心作用是避免因访问不存在的键而引发KeyError错... 目录简介基本语法一、用法二、案例:安全访问未知键三、案例:配置参数默认值简介python是一种高级编

python 常见数学公式函数使用详解(最新推荐)

《python常见数学公式函数使用详解(最新推荐)》文章介绍了Python的数学计算工具,涵盖内置函数、math/cmath标准库及numpy/scipy/sympy第三方库,支持从基础算术到复杂数... 目录python 数学公式与函数大全1. 基本数学运算1.1 算术运算1.2 分数与小数2. 数学函数

python如何调用java的jar包

《python如何调用java的jar包》这篇文章主要为大家详细介绍了python如何调用java的jar包,文中的示例代码简洁易懂,具有一定的借鉴价值,有需要的小伙伴可以参考一下... 目录一、安装包二、使用步骤三、代码演示四、自己写一个jar包五、打包步骤六、方法补充一、安装包pip3 install

Python中help()和dir()函数的使用

《Python中help()和dir()函数的使用》我们经常需要查看某个对象(如模块、类、函数等)的属性和方法,Python提供了两个内置函数help()和dir(),它们可以帮助我们快速了解代... 目录1. 引言2. help() 函数2.1 作用2.2 使用方法2.3 示例(1) 查看内置函数的帮助(