Java解决递归造成的堆栈溢出问题
在Java中,递归造成的堆栈溢出问题通常是因为递归调用的深度过大,导致调用栈空间不足。解决这类问题的一种常见方法是使用非递归的方式重写算法,即使用迭代替代递归。
1.方法一:非递归的方式重写算法(迭代替代递归)
下面通过一个典型的递归例子——计算斐波那契数列的第n项,来演示如何用迭代的方式避免堆栈溢出。
1.1递归版本的斐波那契数列
递归版本的斐波那契数列实现很简单,但是效率较低,尤其是对于大的n值,很容易造成堆栈溢出。
public class FibonacciRecursive {
public static int fibonacci(int n) {
if (n <= 1) {
return n;
} else {
return fibonacci(n - 1) + fibonacci(n - 2);
}
}
public static void main(String[] args) {
int n = 40; // 尝试较大的数,比如40,可能会导致堆栈溢出
System.out.println("Fibonacci(" + n + ") = " + fibonacci(n));
}
}
1.2迭代版本的斐波那契数列
迭代版本的斐波那契数列避免了递归调用,因此不会造成堆栈溢出。
public class FibonacciIterative {
public static int fibonacci(int n) {
if (n <= 1) {
return n;
}
int a = 0, b = 1;
for (int i = 2; i <= n; i++) {
int temp = a + b;
a = b;
b = temp;
}
return b;
}
public static void main(String[] args) {
int n = 90; // 即使n很大,也不会导致堆栈溢出
System.out.println("Fibonacci(" + n + ") = " + fibonacci(n));
}
}
在迭代版本中,我们使用了两个变量a和b来保存斐波那契数列中的连续两个数,通过循环来计算第n项的值。这种方法避免了递归调用,因此不会造成堆栈溢出,即使n的值很大。
1.3小结
通过迭代替代递归是解决递归造成的堆栈溢出问题的常用方法。在实际开发中,如果递归深度可能非常大,建议首先考虑使用迭代的方式来实现算法。
2.方法二:尾递归优化
尾递归是一种特殊的递归形式,递归调用是函数的最后一个操作。在支持尾递归优化的编程语言中(如Scala、Kotlin的某些情况下,以及通过编译器优化或特定设置的Java),尾递归可以被编译器优化成迭代形式,从而避免堆栈溢出。
然而,标准的Java编译器并不自动进行尾递归优化。但是,我们可以手动将递归函数改写为尾递归形式,并使用循环来模拟递归调用栈。
以下是一个尾递归优化的斐波那契数列示例,但请注意,Java标准编译器不会优化此代码,所以这里只是展示尾递归的形式。实际上,要避免Java中的堆栈溢出,还是需要手动将其改写为迭代形式或使用其他技术。
public class FibonacciTailRecursive {
public static int fibonacci(int n, int a, int b) {
if (n == 0) return a;
if (n == 1) return b;
return fibonacci(n - 1, b, a + b); // 尾递归调用
}
public static void main(String[] args) {
int n = 40; // 在标准Java中,这仍然可能导致堆栈溢出
System.out.println("Fibonacci(" + n + ") = " + fibonacci(n, 0, 1));
}
}
实际上,在Java中避免堆栈溢出的正确方法是使用迭代,如之前所示。
3.方法三:使用自定义的栈结构
另一种方法是使用自定义的栈结构来模拟递归过程。这种方法允许你控制栈的大小,并在需要时增加栈空间。然而,这通常比简单的迭代更复杂,且不太常用。
以下是一个使用自定义栈来计算斐波那契数列的示例:
import java.util.Stack;
public class FibonacciWithStack {
static class Pair {
int n;
int value; // 用于存储已计算的值,以避免重复计算
Pair(int n, int value) {
this.n = n;
this.value = value;
}
}
public static int fibonacci(int n) {
Stack<Pair> stack = new Stack<>();
stack.push(new Pair(n, -1)); // -1 表示值尚未计算
while (!stack.isEmpty()) {
Pair pair = stack.pop();
int currentN = pair.n;
int currentValue = pair.value;
if (currentValue != -1) {
// 如果值已经计算过,则直接使用
continue;
}
if (currentN <= 1) {
// 基本情况
currentValue = currentN;
} else {
// 递归情况,将更小的n值压入栈中
stack.push(new Pair(currentN - 1, -1));
stack.push(new Pair(currentN - 2, -1));
}
// 存储计算过的值,以便后续使用
stack.push(new Pair(currentN, currentValue));
}
// 栈底元素存储了最终结果
return stack.peek().value;
}
public static void main(String[] args) {
int n = 40;
System.out.println("Fibonacci(" + n + ") = " + fibonacci(n));
}
}
在这个示例中,我们使用了一个栈来模拟递归过程。每个Pair对象都存储了一个n值和一个对应的斐波那契数值(如果已计算的话)。我们通过将较小的n值压入栈中来模拟递归调用,并在需要时从栈中取出它们来计算对应的斐波那契数值。这种方法允许我们控制栈的使用,并避免了递归造成的堆栈溢出问题。
Java解决递归造成的堆栈溢出问题的更多相关文章
- 使用es6的蹦床函数解决递归造成的堆栈溢出
首先,我们先定义一个函数,使用递归的思想写求和的方法: function sum(x, y) { if (y > 0) { return sum(x + 1, y - 1); } else ...
- Javascript中递归造成的堆栈溢出及解决方案
关于堆栈的溢出问题,在Javascript日常开发中很常见,Google了下,相关问题还是比较多的.本文旨在描述如何解决此类问题. 首先看一个实例(当然你可以使用更容易的方式实现,这里我们仅探讨递归) ...
- python递归次数和堆栈溢出问题
在做递归的时候,测试了一下python的递归能力. 如果不设置递归次数的话,大概只能在992次左右,就会出现错误:RuntimeError: maximum recursion depth excee ...
- javascript递归导致的堆栈溢出
function foo() {foo(); //setTimeout(foo, 0); } foo() 原因是每次执行代码时,都会分配一定尺寸的栈空间(Windows系统中为1M),每次方法调用 ...
- JPA 一对多双向映射 结果对象相互迭代 造成堆栈溢出问题方法
问题: JPA 在双向映射时,会相互包含对方的实例,相互引用,造成递归迭代,堆栈溢出(java.lang.StackOverflowError). 分析: 在后端向前端传递的时候会将数据序列化,转为j ...
- 如何解决js递归里面出现的堆栈溢出
16.下面的递归代码在数组列表偏大的情况下会导致堆栈溢出.在保留递归模式的基础上,你怎么解决这个问题? var list = readHugeList(); var nextListItem = fu ...
- 【Android】Java堆栈溢出的解决办法
分类:C#.Android.VS2015: 创建日期:2016-03-18 随着项目中添加的.jar和.so文件越来越多,编译MyDemos项目时,可能会出现Java堆栈溢出的错误,提示让增加Java ...
- Java常见的几种内存溢出及解决方法
Java常见的几种内存溢出及解决方法[情况一]:java.lang.OutOfMemoryError:Javaheapspace:这种是java堆内存不够,一个原因是真不够(如递归的层数太多等),另一 ...
- 前端知识体系:JavaScript基础-作用域和闭包-闭包的实现原理和作用以及堆栈溢出和内存泄漏原理和相应解决办法
闭包的实现原理和作用 闭包: 有权访问另一个函数作用域中的变量的函数. 创建闭包的常见方式就是,在一个函数中创建另一个函数. 闭包的作用: 访问函数内部变量.保持函数在环境中一直存在,不会被垃圾回收机 ...
- 深入理解java虚拟机---java内存区域与内存溢出异常---2堆栈溢出
本文来源于翁舒航的博客,点击即可跳转原文观看!!!(被转载或者拷贝走的内容可能缺失图片.视频等原文的内容) 若网站将链接屏蔽,可直接拷贝原文链接到地址栏跳转观看,原文链接:https://www.cn ...
随机推荐
- 基于SDF的光照效果
基于SDF的光照效果 好久没写博客了,怠惰了,就当爬了一步 原神二次元风格面部渲染 效果 Show me the code Shader "Unlit/SDF" { Propert ...
- WPF Canvas在Image 图像上绘图,自适应缩放.
效果如图 实现了绘图,自适应缩放 核心代码如下 <Window.InputBindings> <KeyBinding Key="Z" Modifiers=&quo ...
- python 第一次主要是if while 奇偶数的练习
例如输出1-10的数字,但是7除外. a=1 while a < 11: if a == 7: pass else: print(a) a=a+1 输出1-100所以的偶数 a=1 while ...
- 高通参考设计中MTP与QRD
高通参考设计中MTP与QRD 背景 之前在调试设备树的时候,看到设备树带了一个qrd的后缀,一直没搞清楚.上网找资料也好像不是我想要的. 今天查阅lk侧的代码,发现了HW_PLATFORM_HRD这个 ...
- 🚀 Karpor - 让 AI 全面赋能 Kubernetes!
什么是 Karpor? 一言以蔽之,Karpor 是一个现代化的 Kubernetes 可视化工具,核心特性聚焦在 搜索. 洞察. AI ,目标是更方便快捷地连接平台和多集群,并用 AI 赋能 Ku ...
- 在SelfHost项目中获取客户端IP地址
在SelfHost项目中,获取客户端的IP地址比OwinSelfHost项目要复杂一些,可以通过以下方法获得: base.Request.Properties["System.Service ...
- Redis 高阶应用
生成全局唯一 ID 全局唯一 ID 需要满足以下要求: 唯一性:在分布式环境中,要全局唯一 高可用:在高并发情况下保证可用性 高性能:在高并发情况下生成 ID 的速度必须要快,不能花费太长时间 递增性 ...
- JavaScript --函数--手稿
- 阿里云Centos7 安装mysql5.7 报错:./mysqld: error while loading shared libraries: libaio.so.1: cannot open shared object file: No such file or directory
在阿里云服务器Centos7中安装mysql5.7,解压数据库初始化后,报错 ./mysqld: error while loading shared libraries: libaio.so.1: ...
- 全网最适合入门的面向对象编程教程:04 类和对象的 Python 实现-为自定义类添加方法(PySerial 库接收串口数据)
全网最适合入门的面向对象编程教程:04 类和对象的 Python 实现-为自定义类添加方法(PySerial 库接收串口数据) 摘要: 本文我们主要讲解了如何为自定义类添加方法,pyseria 库的基 ...