什么鬼?又发生StackOverflowError?
一、栈内存溢出(StackOverflowError)?栈是线程私有的,他的生命周期与线程相同,每个方法在执行的时候都会创建一个栈帧,用来存储局部变 Java栈,是指Java虚拟机栈,本...
·
一、栈内存溢出(StackOverflowError)?
栈是线程私有的,他的生命周期与线程相同,每个方法在执行的时候都会创建一个栈帧,用来存储局部变 Java栈,是指Java虚拟机栈,本地方法栈;
Java虚拟机栈是Java方法执行的内存模型,每个方法执行的同时都会创建一个栈帧用于存储局部变量,操作数栈,动态链接(在一个方法中调用另一个方法),方法出口等信息。每一个方法从调用直至执行完成的过程,就对应一个栈帧在虚拟机栈中入栈和出栈的过程,生命周期与线程相同。
之前一直说的栈实际就是Java虚拟机栈,更详细一点,就是Java虚拟机栈中的局部变量部分,存放堆中对象的地址;
本地方法栈和Java虚拟机栈的作用完全一样,区别:本地方法栈为虚拟机使用的Native方法服务,虚拟机栈是为了JVM执行的java方法服务;
那么显然,如果入栈太多,大过虚拟机所运行的最大深度,就会抛出StcakOverFlow异常
二、模拟栈内存溢出
那么我们来模拟一下,可以直接想到的就是递归,递归调用方法,且无出口。
同样,为了尽快看到效果,我们可以设置一下栈的深度,打开Run/Debug Configuration ,通过-Xss来设置,这里设置为256kb;
代码如下:我们设置一个私有变量i来反馈调用栈的深度
package cn.liuhaihua;
import java.util.ArrayList;
import java.util.List;
public class StackOverflow {
private static int i=0;
static List<Student> list = new ArrayList<Student>();
public static void main(String[] args)throws Exception {
int i=1;
test();
}
public static void test(){
i++;
System.out.println(i);
while (true) {
test();
}
}
}
运行结果
*** java.lang.instrument ASSERTION FAILED ***: "!errorOutstanding" with message transform method call failed at JPLISAgent.c line: 844
*** java.lang.instrument ASSERTION FAILED ***: "!errorOutstanding" with message transform method call failed at JPLISAgent.c line: 844
Exception in thread "main" java.lang.StackOverflowError
at sun.nio.cs.UTF_8.updatePositions(UTF_8.java:77)
at sun.nio.cs.UTF_8.access$200(UTF_8.java:57)
at sun.nio.cs.UTF_8$Encoder.encodeArrayLoop(UTF_8.java:636)
at sun.nio.cs.UTF_8$Encoder.encodeLoop(UTF_8.java:691)
at java.nio.charset.CharsetEncoder.encode(CharsetEncoder.java:579)
at sun.nio.cs.StreamEncoder.implWrite(StreamEncoder.java:271)
三、如何避免?
1.避免程序发生递归,特别是没有加退出条件的递归。一定慎重
2.合理设置栈的大小。
更多推荐
已为社区贡献2条内容
所有评论(0)