编写堆溢出和栈溢出程序的目的是更好的理解JVM内存中堆和栈存储的是什么内容。

堆存放对象实例。

栈存放栈帧。

栈溢出,栈存储方法调用时的栈帧,所以可以使用没有退出条件的递归实现:

//栈溢出示例public class StackOverflowTest {    public static void main(String[] args) {        new StackOverflowTest().test();
    }    // 递归调用
    public void test() {
        test();
}

执行结果:

Exception in thread "main" java.lang.StackOverflowError
    at javaEx.StackOverflowTest.test(StackOverflowTest.java:22)

堆中存放对象实例,只要new创建的实例足够大就能溢出

// 堆溢出实例public class HeadOverTest {    public static void main(String[] args) {
     int i = 1;          //记录第几次创建时溢出
     List
        while (true) {
            System.out.println("执行次数:"+(i++));
            list.add(new byte[1024*1024*1024]);
}

执行结果:

执行次数:1执行次数:2执行次数:3Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
    at javaEx.HeadOverTest.main(HeadOverTest.java:17)

特别说明:

这里使用的是List集合存放实例化的byte数组,如果不用List存放直接使用new关键字实例化将无法实现溢出。

因为new实例化没有地方调用会被垃圾回收机制回收,推就很难或不会溢出了

public class HeadOverTest {    public static void main(String[] args) {//        Listlist = new ArrayList<>();
        int i = 1;          //记录第几次创建时溢出
        byte [] bytes;        while (true) {
            System.out.println("执行次数:"+(i++));//            list.add(new byte[1024*1024*1024]);
            bytes = new byte[1024*1024*1024];
}

执行结果:

执行次数:124执行次数:125执行次数:126执行次数:127执行次数:128执行次数:129执行次数:130执行次数:131
根本停不下来。。。。。。。而且风扇狂转,内存占用飙升,CPU占用也会升高