编写堆溢出和栈溢出程序的目的是更好的理解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占用也会升高