JVM中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: 阅读异常堆栈信息

  1. 抛出异常的入口
    at outofmemoryerror.HeapOOM.main(HeapOOM.java:17)
  2. 可以点击(HeapOOM.java:17) 数字到出错行
  3. 然后ctrl + alt + 鼠标左键 组合键点击add 然后找到 实现类·ArrayList; 因为这是父类的引用指向子类的对象,需要找到具体的实现类。
  4. 然后找到了ArrayListadd()方法,依次点击进入
  5. ctrl 再右击 ensureCapacityInternal() 方法
  6. ctrl 再右击 ensureExplicitCapacity() 方法
  7. ctrl 再右击 grow()方法
  8. ctrl 再右击 ArrayscopyOf()方法
  9. ctrl 再右击 ArrayscopyOf()方法。 最后定位到这个地方为出错点,附上代码行。

    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

  1. 点击异常堆栈信息括号后面的数字可以定位到具体抛出异常的位置;
  2. debug断点调试功能也能定位到出错位置

1.4 结果分析

当java堆内存溢出时,异常堆栈信息 java.lang.OutOfMemoryError 后面会跟着 Java heap space

根据堆快照需要找出出错的原因,确认是什么造成这个原因的。


1.5 内存泄漏和内存溢出

定位

查看出现内存泄漏(Memory Leak) 还是内存溢出 (Memory Overflow);

  1. 内存溢出: 通过工具查看泄漏对象到GC Roots引用链; 这样就能找到泄漏对象是通过怎样的路径与GC Roots相关联导致垃圾收集器无法自动回收它们的。
  2. 检测虚拟机堆参数(-Xmx,-Xms)设置,从代码上查看对象的生命周期,减少程序对内存的消耗等。

需要理解的是: 什么是内存溢出,什么是内存泄漏; 至于区别,理解概念就是区别。


1.6 使用visualvm 来分析dump文件

加载dump文件到到软件中,然后进行分析.


(一)加载文件:

文件基本使用


(二)对dump文件进行分析:
这里写图片描述

借助图形:可以看那个对象的数量最多,谁占用内存最多等。这样即可定位出错原因。


后记

  1. JDK1.8 HotSpot
  2. IDEA 工具

参考

  • 《深入理解java虚拟机》
  • 3
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
JVM(Java虚拟机),OOM(Out of Memory)异常通常由以下几个原因引发: 1. 堆内存不足:堆内存是JVM用于分配对象实例的内存区域。如果应用程序创建了大量的对象,并且堆内存无法容纳这些对象,就会导致堆内存不足的OOM异常。 2. 方法区/永久代内存不足:方法区(在Java 8之前称为永久代)用于存储类的元数据、静态变量、常量池等信息。如果应用程序加载了大量的类或者使用了大量的字符串常量,就可能导致方法区/永久代内存不足的OOM异常。 3. 栈内存溢出:栈内存用于保存方法的调用栈和局部变量等信息。如果方法调用层级过深或者方法使用了大量的局部变量,就可能导致栈内存溢出的OOM异常。 4. 本地方法栈溢出:本地方法栈用于执行本地方法(Native Method)。如果本地方法执行时需要消耗大量的本地内存,而本地内存不足,就可能导致本地方法栈溢出的OOM异常。 5. 直接内存溢出:直接内存是JVM使用的一种特殊的堆外内存,它不受JVM堆内存大小限制。如果应用程序频繁地分配大量的直接内存,而没有及时释放,就可能导致直接内存溢出的OOM异常。 需要注意的是,不同的JVM实现可能在内存管理方面存在差异,因此具体的OOM异常可能会有所不同。此外,可以通过调整JVM的启动参数或优化代码来减少OOM异常的发生。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值