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相关的学习心得。

(0)

相关推荐

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

    JVM的框架知识了解之后,实际的项目里发生了OOM异常的话,怎么获取以及分析异常信息后怎么分析呢. 这里稍微做一下归纳. 第一步,首先通过下面两个方法的任何一种,把发生OOM时的heap信息dump下来. 有两个方法,通过设置可以把OOM时的dump信息获取到: 1)方法1:在JVM的启动参数里添加如下命令 -XX:+HeapDumpOnOutOfMemoryError 2)方法2:在JDK的安装目录下,找到bin目录,然后双击执行"jvisualvm.exe" 执行程序之后,在视图里

  • Java在运行时识别类型信息的方法详解

    前言 在日常的学习工作当中,有一些知识是我们在读书的时候就能够习得:但有一些知识不是的,需要在实践的时候才能得到真知--这或许就是王阳明提倡的"知行合一". 在Java中,并不是所有的类型信息都能在编译阶段明确,有一些类型信息需要在运行时才能确定,这种机制被称为RTTI,英文全称为Run-Time Type Identification,即运行时类型识别,有没有一点"知行合一"的味道?运行时类型识别主要由Class类实现. 01 Class类 在Java中,我们常用

  • 超全的iOS各种设备信息获取方法总结(包括iPhone8/iPhone X)

    前言 在开发中,有的时候为了统计用户信息.下发广告,服务器端往往需要手机用户设备及app的各种信息,这些信息的获取可以根据不同的设备或者App.系统版本来提供不同的功能或更好的用户体验,或者让开发者能更好的分析用户的问题原因. 下面讲述一下各种信息的获取方式: 点击下载以上展示效果的GitHub源码 ,大家也可以 本地下载 一行代码就搞定的统一来! // 这个方法后面会列出来 NSString *deviceName = [self getDeviceName]; NSLog(@"设备型号--&

  • 解决jmap命令打印JVM堆信息异常的问题

    jmap命令可以打印java进程的JVM堆信息,今天在某台机器上运行该命令查看 19560进程的堆信息 jmap -heap 19560 出现以下异常 Attaching to process ID 19560, please wait... Debugger attached successfully. Server compiler detected. JVM version is 24.79-b02 using thread-local object allocation. Paralle

  • JVM 心得分享(加载 链接 初始化)

    基本概念:类加载的过程大致分为三个阶段 1.加载阶段:本阶段主要把class的二进制代码加载进入JVM,并且进行常量池(类名,方法名,字段名),方法区(二进制字节码),栈(本地方法栈结构),堆(java.lang.class对象)的设置. 有三个加载类:Bootstrap ClassLoader,加载jre/lib/下的类: Extension ClassLoader:加载jre/lib/ext下的类: ApplicationClassLoader:加载classpath下的类(应用程序自己开发

  • JVM入门之内存结构(堆、方法区)

    目录 1.堆 1.1 定义 1.2 堆的作用 1.3 特点 1.4 堆内存溢出 1.5 堆内存诊断 2.方法区 2.1 结构(1.6 对比 1.8) 2.2 内存溢出 2.3 常量池 2.4 运行时常量池 2.5 常量池与串池的关系 2.6 StringTable的位置 2.7 StringTable 垃圾回收 2.8 方法区的垃圾回收 3.直接内存 释放原理 1.堆 1.1 定义 是Java内存区域中一块用来存放对象实例的区域[几乎所有的对象实例都在这里分配内存] 通过new关键字创建的对象都

  • Java获取代码中方法参数名信息的方法

    前言 大家都知道随着java8的使用,在相应的方法签名中增加了新的对象Parameter,用于表示特定的参数信息,通过它的getName可以获取相应的参数名.即像在代码中编写的,如命名为username,那么在前台进行传参时,即不需要再编写如@Parameter("username")类的注解,而直接就能进行按名映射. 如下的代码参考所示: public class T { private interface T2 { void method(String username, Stri

  • Java实现获取cpu、内存、硬盘、网络等信息的方法示例

    本文实例讲述了Java实现获取cpu.内存.硬盘.网络等信息的方法.分享给大家供大家参考,具体如下: 1. 下载安装sigar-1.6.4.zip 使用java自带的包获取系统数据,容易找不到包,尤其是内存信息不够准确,所以选择使用sigar获取系统信息. 下载地址:http://sourceforge.net/projects/sigar/files/latest/download?source=files 或点击此处本站下载. 解压压缩包,将lib下sigar.jar导入eclipse的CL

  • 在Linux与Windows上获取当前堆栈信息的方法

    在编写稳定可靠的软件服务时经常用到输出堆栈信息,以便用户/开发者获取准确的运行信息.常用在日志输出,错误报告,异常检测. 在Linux有比较简便的函数获取堆栈信息: #include <stdio.h> #include <execinfo.h> #include <signal.h> #include <stdlib.h> #include <unistd.h> void handler(int sig) { void *array[5]; s

  • C#获取程序文件相关信息的方法

    本文实例讲述了C#获取程序文件相关信息的方法,分享给大家供大家参考. 具体实现方法如下: using System.Reflection; using System.Runtime.CompilerServices; // // 有关程序集的常规信息是通过下列 // 属性集控制的.更改这些属性值可修改与程序集 // 关联的信息. // [assembly: AssemblyTitle("")] [assembly: AssemblyDescription("")]

随机推荐