Bootstrap

常见的栈溢出StackOverFlow 与 内存溢出OutOfMemory的区别

0、前言:内存模型

        对于多线程运行情况下的jvm内存,我们应当知道:

        每创建一个线程,jvm就会为其分配一块线程私有的工作内存,其中包括程序计数器、栈,等等。
        对于每一个线程私有的栈,当线程执行方法时,栈内都会分配一块栈帧,用于保存方法的局部变量表、方法返回地址等信息。
                                                                                        ——详细见《深入理解java虚拟机》

概括地看,就是说:
        线程会占用“总的内存”、运行方法所创建的栈帧会占用“栈的内存”

以下内容都可以结合前言这一节来理解:

1、StackOverFlow 

         栈溢出,顾名思义,栈爆掉了,栈为什么会爆掉?因为里面的栈帧太多了造成的呗!

         如果不断地调用方法,不断地往一个栈里压入栈帧,就会造成StackOverFlow,例如,让同一个线程递归地调用方法:

        在每次递归调用时,JVM会将当前方法的局部变量、参数和返回地址等信息保存在栈帧中,并将新的栈帧推入栈中。
        当递归的层数过多时,就会产生大量的栈帧,占用过多的栈内存空间。如果超过了栈的容量限制,就会发生栈溢出,抛出StackOverflowError异常。

代码示例:

public class RecursionExample {
    public static void recursiveMethod() {
        recursiveMethod(); // 递归调用
    }

    public static void main(String[] args) {
        try {
            recursiveMethod();
        } catch (StackOverflowError e) {
            System.out.println("StackOverflowError: " + e.getMessage());
        }
    }
}

 2、OutOfMemory(OOM)

        内存溢出,顾名思义,内存爆掉了。为什么内存会爆掉呢?那就有可能是栈创建太多了!

        对比前文,例如,如果应用程序中创建了大量线程,并且每个线程都占用了一定的内存资源,那么JVM中可用的内存可能会被耗尽,导致OutOfMemoryError。

        此外,内存泄漏也有可能导致OutOfMemoryError:
        应用程序中的对象无法被垃圾回收器正确释放时,会导致内存泄漏。这可能是因为持有对对象的引用,并且这些对象不再需要,但仍然存在于内存中。随着时间的推移,内存中的对象越来越多,最终耗尽了可用的内存资源。

3、总结

        从总体角度看,栈是在内存里面的。
        栈溢出就是栈满了,内存溢出就是内存满了。
        引用另一篇文章http://t.csdn.cn/t1Bd1里的一句话:

        可以把内存比作是一个大箱子,栈是一个小箱子,栈溢出是指小箱子装不下了;而栈内存溢出是大箱子在也装不下小箱子了

;