JVM 心得 OOM时的堆信息获取方法与分析

 更新时间:2017年10月17日 09:36:54   作者:Michael_Chong  
下面小编就为大家带来一篇JVM 心得 OOM时的堆信息获取方法与分析。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧

JVM的框架知识了解之后,实际的项目里发生了OOM异常的话,怎么获取以及分析异常信息后怎么分析呢。

这里稍微做一下归纳。

第一步,首先通过下面两个方法的任何一种,把发生OOM时的heap信息dump下来。

有两个方法,通过设置可以把OOM时的dump信息获取到:

1)方法1:在JVM的启动参数里添加如下命令

-XX:+HeapDumpOnOutOfMemoryError

2)方法2:在JDK的安装目录下,找到bin目录,然后双击执行"jvisualvm.exe"

执行程序之后,在视图里可以看到正在执行的java程序和Java线程ID(PID:xxx)。

右键选中"在OOM时生成dump"。

第二步,执行如下代码(本例使用方法1。如果使用方法2时需要先执行代码,再在JvirtualVM中选中java程序设置OOM时生成dump):

代码的意义是每一次创建一个1*1024*1024大小的int数组。在While中循环,引起Heap的OutOfMemory异常发生。

有一个小知识点共享下,map进行containsKey的处理时,key如果是javabean对象,判断时使用的是对象的引用。所以每次判断都是新的对象。

最好key的数据类型定义为值类型等(String作为key时使用常量字符串对String赋值如String key ="key",不要用String key = new String(“key”)的形式,这样又变成了一个引用对象了)。

package com.chong.studyparalell.jvm.heap;

import java.util.HashMap;
import java.util.Map;

public class HeapLeakage {

static class MapKey{
Integer key;
MapKey(Integer key){
this.key = key;
}
}

public static void main(String []args){

Map<MapKey,Integer[]> localMap = new HashMap();
while(true){
for(int i = 0;i<10;i++){
try{
Thread.sleep(100);
}catch(Exception e){
e.printStackTrace();
}
MapKey mapKey = new MapKey(new Integer(i));
if(!localMap.containsKey(mapKey)){
localMap.put(mapKey, new Integer[1*1024*1024]);
}
}
}
}
}

日志如下:

JAVAGENT: call premain instrumentation for class SizeOf
java.lang.OutOfMemoryError: Java heap space
Dumping heap to java_pid31512.hprof ...
Heap dump file created [491663234 bytes in 0.930 secs]
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
at com.chong.studyparalell.jvm.heap.HeapLeakage.main(HeapLeakage.java:38)

第三步,在项目的classpath目录下确认"java_pidXXX.hprof"文件是否存在,正常情况下应该存在的。

在jvirtualvm中,点击"文件"->"装入"->选中上述hprof文件。

点击“类”tab页,通过"大小"排序,可以看出java.lang.Integer占用了99%以上的空间。

双击这一行,可以进入进去看到,各个Integer的具体内容。这里的目的是发生OOM,所以Integer的内容没有设值,进去可以发现值都是null。

那么实际的项目里,可以观看发生问题线程里哪些对象的一直是活着的,并且size远远超出预期,重点分析这些可疑对象。推测的方向:是否在循环里大量的实例化对象,加载DB数据时一次加载了大量的数据,以及是否发生了内存泄露(如长生命周期的Map和Set,List中一直保存在大量的不再使用的对象引用。),以及静态变量的使用等。

后记:

通过这几天的JVM复习,以及OOM发生时的dump分析,再次做项目时,头脑会更清晰一些。^-^
JVM告一段落了,接下来会发一些跟SpringBoot和Redis相关的学习心得。

相关文章

  • Java接口和抽象类有什么区别

    Java接口和抽象类有什么区别

    这篇文章主要介绍了Java接口和抽象类有什么区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-12-12
  • Java中输入与输出的方法总结

    Java中输入与输出的方法总结

    这篇文章主要为大家总结了一下Java中输入与输出的三种方法,文中通过示例详细的讲解了一下这些方法的使用,需要的小伙伴可以参考一下
    2022-04-04
  • RocketMQ的消费者类型与最佳实践详解

    RocketMQ的消费者类型与最佳实践详解

    这篇文章主要介绍了RocketMQ的消费者类型与最佳实践详解,在 RocketMQ 5.0 中,更加强调了客户端类型的概念,尤其是消费者类型,为了满足多样的 RocketMQ 中一共有三种不同的消费者类型,分别是 PushConsumer、SimpleConsumer 和 PullConsumer,需要的朋友可以参考下
    2023-10-10
  • 在IDEA中搭建最小可用SpringMVC项目(纯Java配置)

    在IDEA中搭建最小可用SpringMVC项目(纯Java配置)

    这篇文章主要介绍了在IDEA中搭建最小可用SpringMVC项目(纯Java配置),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-12-12
  • LoggingEventAsyncDisruptorAppender类执行流程源码解读

    LoggingEventAsyncDisruptorAppender类执行流程源码解读

    这篇文章主要介绍了LoggingEventAsyncDisruptorAppender类执行流程源码解读,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-12-12
  • JVM的7种垃圾回收器(小结)

    JVM的7种垃圾回收器(小结)

    这篇文章主要介绍了JVM的7种垃圾回收器(小结),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-10-10
  • springboot项目突然启动缓慢的解决

    springboot项目突然启动缓慢的解决

    这篇文章主要介绍了springboot项目突然启动缓慢的解决,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • java设计模式—静态代理模式(聚合与继承方式对比)

    java设计模式—静态代理模式(聚合与继承方式对比)

    下面小编就为大家带来一篇java设计模式—静态代理模式(聚合与继承方式对比)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-05-05
  • SpringBoot快速实现接口消息加密的过程详解

    SpringBoot快速实现接口消息加密的过程详解

    在项目中,为了保证数据的安全,我们常常会对传递的数据进行加密,常用的加密算法包括对称加密(AES)和非对称加密(RSA),博主选取码云上最简单的API加密项目进行下面的讲解,需要的朋友可以参考下
    2023-11-11
  • java入门概念个人理解之package与import浅析

    java入门概念个人理解之package与import浅析

    下面小编就为大家带来一篇java入门概念个人理解之package与import浅析。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-08-08

最新评论