由于面试官仅提到OOM,但 Java 的OOM又分很多类型的呀:
- 堆溢出(“java.lang.OutOfMemoryError: Java heap space”)
- 永久代溢出(“java.lang.OutOfMemoryError:Permgen space”)
- 不能创建线程(“java.lang.OutOfMemoryError:Unable to create new native thread”)
- OOM在《Java虚拟机规范》里,除程序计数器,虚拟机内存的其他几个运行时区域都可能发生OOM,那本文的目的是啥呢?
- 通过代码验证《Java虚拟机规范》中描述的各个运行时区域储存的内容
- 在工作中遇到实际的内存溢出异常时,能根据异常的提示信息迅速得知是哪个区域的内存溢出,知道怎样的代码可能会导致这些区域内存溢出,以及出现这些异常后该如何处理。
本文代码均由笔者在基于OpenJDK 8中的HotSpot虚拟机上进行过实际测试。
01 Java堆溢出
Java堆用于储存对象实例,只要不断地创建对象,并且保证GC Roots到对象之间有可达路径来避免GC机制清除这些对象,则随对象数量增加,总容量触及最大堆的容量限制后就会产生内存溢出异常。
限制Java堆的大小20MB,不可扩展
-XX:+HeapDumpOnOutOf-MemoryError
可以让虚拟机在出现内存溢出异常的时候Dump出当前的内存堆转储快照。
案例1
- 报错
Java堆内存的OOM是实际应用中最常见的内存溢出异常场景。出现Java堆内存溢出时,异常堆栈信息“
java.lang.OutOfMemoryError”会跟随进一步提示“Java heap space”。
那既然发生了,如何解决这个内存区域的异常呢</