前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >JVM中OutOfMemoryError异常案例一之堆溢出

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

作者头像
用户7886150
修改2020-12-15 10:38:07
3780
修改2020-12-15 10:38:07
举报
文章被收录于专栏:bit哲学院

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

本文系转载,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文系转载前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档