编写堆溢出和栈溢出程序的目的是更好的理解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<byte[]> list = new ArrayList<>();
while (true) { System.out.println("执行次数:"+(i++)); list.add(new byte[1024*1024*1024]); } } }
执行结果:
执行次数:1 执行次数:2 执行次数:3 Exception 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) { // List<byte[]> list = 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占用也会升高