java.lang.StackOverflowError堆栈溢出错误

链接1:https://blog.csdn.net/zc375039901/article/details/79179465

链接2:http://www.360doc.com/content/14/0925/13/1073512_412236522.shtml

提问:为什么程序会出现StackOverflowError异常,为什么别人老是让菜鸟少用递归。

出现原因: 不停的循环递归调用

 

模拟堆栈溢出:

public class JavaStatck{

private static int index = 1;

private void method() {
    index++;
    method();
}

public static void main(String[] args) {
    JavaStatck statck = new JavaStatck();
    try {
        statck.method();
    } catch (Throwable e) {
        System.out.println("栈深度: " + index);
        e.printStackTrace();
    }
}

}

分析原因:

    执行main函数会创建一个线程,同时创建一个虚拟机栈(栈内存),这时里面会保持一个栈帧(Stack Frame)用于存放method() 运行期数据的数据集(存放基本数据类型变量、引用类型的变量、returnAddress类型的变量、操作数栈、动态链接、方法出口信息等),如果main方法里面调用多个方法,就会有多个栈帧。

当调用stack.method()时,会产生一个栈帧区块,每次method()递归调用时,都会产生一个新的栈帧区块,这时就会不停产生新的栈帧区块,当栈内存超过系统配置的栈内存,就会出现java.lang.StackOverflowError异常。

 

基础知识:

     虚拟机栈:栈也叫栈内存,是java虚拟机的内存模型之一。它的生命周期是在线程创建时创建,线程结束而消亡,释放内存。因此是私有的,不可共享。栈存储的数据,以栈帧为单位存储,栈帧是一个内存区块,是一个数据集,是一个有关方法和运行期数据的数据集,当一个方法A被调用时就产生了一个栈帧F1,并被压入到栈中,A方法又调用了B方法,于是产生栈帧F2也被压入栈,B方法执行完毕后,F2栈帧先出栈,F1栈帧后出栈,遵循“先进后出”原则。

      栈帧:没当一个java方法被执行时都会在虚拟机中新创建一个栈帧,方法调用结束后即被销毁。


栈帧存储数据包含以下5个部分:

1、局部变量表:保存函数的参数以及局部变量用的,局部变量表中的变量只在当前函数调用中生效,当函数调用结束后,随着函数栈帧的销毁,局部变量表也随之销毁。

存放基本数据类型变量(Boolean、byte、char、short、int、float)、引用类型的变量(reference)、returnAddress(指向一条字节码指令的地址)类型的变量。

2、操作数栈:主要用于保存计算过程的中间结果,同时作为计算过程中变量临时的存储空间。只支持出栈入栈操作。在概念模型中,两个栈帧是相互独立的。但是大多数虚拟机的实现都会进行优化,令两个栈帧出现一部分重叠。令下面的部分操作数栈与上面的局部变量表重叠在一块,这样在方法调用的时候可以共用一部分数据,无需进行额外的参数复制传递。

3、动态链接

4、方法出口信息

5、其他

posted @ 2018-03-28 15:15  人类能走多远  阅读(327)  评论(0)    收藏  举报