JVM中OutOfMemoryError异常案例一之堆溢出

2020-12-15 10:38:07 浏览数 (1)

参考链接: 了解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虚拟机》

0 人点赞