之前总结JVM内存区域时提及相关区域及发生在其上的OutOfMemoryError异常,下面将依照除程序计数器(这个区域并没有规定OOM异常)外发生OOM的相关原因、示例代码和解决方法,为以后遇到相关问题时解决提供简单的预习。
实战:OOM异常
在《Java虚拟机规范》中描述了两种异常:
1)如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出SOF异常。
2)如果虚拟机的占内存允许动态扩展,当可以栈占内存无法申请到足够的内存时,将抛出OOM异常。
But,《规范》中明确允许Java虚拟机实现自行选择是否支持站的动态扩展,而Oracle目前使用的HotSpot的选择是不可拓展,所以除非在创建线程申请内存是就因无法获得足够的内存而出现OOM异常,否则在线程运行时是不会因为扩展而导致内存溢出,只会因为栈容量无法容纳新的栈帧而导致SOF异常
实例代码中注释了执行时需要设置的虚拟机参数(注释中“VM Args”后面跟着的参数),这些参数对实验的结果有直接影响。
1、Java堆溢出
Java堆中存储对象实例,当GC Roots到对象的可用路径避免垃圾回收机制清除对象时,只要不断创建对象,随着对象的增加,总容量触及最大堆内存容量限制后将产生内存溢出异常(OOM Error)。
以下代码规定了一个不可拓展的大小为20MB的Java堆(调整堆的最小值-Xms参数与最大值-Xmx参数甚至为一样即可避免对自动拓展),通过参数-XX:+HeapDumpOnOutOf-MemoryError可以让虚拟机在出现内存溢出异常的时候Dump出当前的内存堆转储快照以便进行事后分析。
Java堆内存溢出异常异常测试:
/**
* VM Args:-Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError
* @author zzm
*/
public class Demo001 {
static class OOMObject {
}
public static void main(String[] args) {
List<OOMObject> list = new ArrayList<OOMObject>();
while (true) {
list.add(new OOMObject());
}
}
}
运行结果:
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
要解决这个内存区域的异常,常规的处理方法是首先通过内存映像分析工具(如Eclipse MemoryAnalyzer)对Dump出来的堆转储快照进行分析。第一步首先应确认内存中导致OOM的对象是否是必要的,也就是要先分清楚到底是出现了内存泄漏(Memory Leak)还是内存溢出(M