参考链接: 了解Java中的OutOfMemoryError异常
JVM中OutOfMemoryError异常案例一之堆溢出
介绍java堆的OOM
1. java堆溢出
Java堆存储对象实例,只要不断创建对象,并且保证GC Roots到对象之间有可达路径来避免垃圾回收机制清除这些对象,那么当对象到达堆内存允许的最大容量时就会报异常。
1.1. 参数设置
-XX: HeapDumpOnOutOfMemoryError
-XX:HeapDumpPath=d:/jvmdump/HeapOOM.dump
-Xms20M
-Xmx20M
-XX: PrintGCDetails
-Xms : 初始堆大小; -Xms20m:表示初始堆20M -Xmx : 最大堆大小; -Xmx20m : 表示最大可用20M HeapDumpOnOutOfMemoryError: 将溢出转存dump快照 -XX:HeapDumpPath :转存的dump快照通常都需要指定一个路径。然后分析结果 表示使用 - 表示不使用
1.2 源码
public class HeapOOM {
static class OOMObject {
}
public static void main(String[] args) {
List<OOMObject> list = new ArrayList<OOMObject>();
while (true) {
list.add(new OOMObject());
}
}
}
步骤操作:
使用IDEA,如下进行参数设置:
1.3 结果
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
at java.util.Arrays.copyOf(Arrays.java:3210)
at java.util.Arrays.copyOf(Arrays.java:3181)
at java.util.ArrayList.grow(ArrayList.java:261)
at java.util.ArrayList.ensureExplicitCapacity(ArrayList.java:235)
at java.util.ArrayList.ensureCapacityInternal(ArrayList.java:227)
at java.util.ArrayList.add(ArrayList.java:458)
at outofmemoryerror.HeapOOM.main(HeapOOM.java:17)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:147)
额外补充1: 阅读异常堆栈信息
抛出异常的入口 at outofmemoryerror.HeapOOM.main(HeapOOM.java:17)可以点击(HeapOOM.java:17) 数字到出错行然后ctrl alt 鼠标左键 组合键点击add 然后找到 实现类·ArrayList; 因为这是父类的引用指向子类的对象,需要找到具体的实现类。然后找到了ArrayList的 add()方法,依次点击进入按ctrl 再右击 ensureCapacityInternal() 方法按ctrl 再右击 ensureExplicitCapacity() 方法按ctrl 再右击 grow()方法按ctrl 再右击 Arrays 的 copyOf()方法按ctrl 再右击 Arrays 的copyOf()方法。 最后定位到这个地方为出错点,附上代码行。
public static <T,U> T[] copyOf(U[] original, int newLength, Class<? extends T[]> newType) {
@SuppressWarnings("unchecked")
T[] copy = ((Object)newType == (Object)Object[].class)
? (T[]) new Object[newLength]
: (T[]) Array.newInstance(newType.getComponentType(), newLength);
System.arraycopy(original, 0, copy, 0,
Math.min(original.length, newLength));
return copy;
}
原因:数组不能继续在堆中申请到额外的内存空间而抛出内存溢出。
额外补充2
点击异常堆栈信息括号后面的数字可以定位到具体抛出异常的位置; debug断点调试功能也能定位到出错位置
1.4 结果分析
当java堆内存溢出时,异常堆栈信息 java.lang.OutOfMemoryError 后面会跟着 Java heap space
根据堆快照需要找出出错的原因,确认是什么造成这个原因的。
1.5 内存泄漏和内存溢出
定位:
查看出现内存泄漏(Memory Leak) 还是内存溢出 (Memory Overflow);
内存溢出: 通过工具查看泄漏对象到GC Roots的引用链; 这样就能找到泄漏对象是通过怎样的路径与GC Roots相关联导致垃圾收集器无法自动回收它们的。检测虚拟机堆参数(-Xmx,-Xms)设置,从代码上查看对象的生命周期,减少程序对内存的消耗等。
需要理解的是: 什么是内存溢出,什么是内存泄漏; 至于区别,理解概念就是区别。
1.6 使用visualvm 来分析dump文件
加载dump文件到到软件中,然后进行分析.
(一)加载文件:
(二)对dump文件进行分析:
借助图形:可以看那个对象的数量最多,谁占用内存最多等。这样即可定位出错原因。
后记
JDK1.8 HotSpotIDEA 工具
参考
《深入理解java虚拟机》