浅谈JVM之使用JFR解决内存泄露

作者:flydean 时间:2022-10-19 01:55:30 

简介

虽然java有自动化的GC,但是还会有内存泄露的情况。当然java中的内存泄露跟C++中的泄露不同。

在C++中所有被分配的内存对象都需要要程序员手动释放。但是在java中并不需要这个过程,一切都是由GC来自动完成的。那么是不是java中就没有内存泄露了呢?

要回答这个问题我们首先需要界定一下什么是内存泄露。如果说有时候我们不再使用的对象却不能被GC释放的话,那么就可以说发生了内存泄露。

一个内存泄露的例子

我们举一个内存泄露的例子,先定义一个大对象:


public class KeyObject {
   List<String> list = new ArrayList<>(200);
}

然后使用它:


public class TestMemoryLeak {

public static HashSet<Object> hashSet= new HashSet();

public static void main(String[] args) throws InterruptedException {
       boolean flag= true;
       while(flag){
           KeyObject keyObject= new KeyObject();
           hashSet.add(keyObject);
           keyObject=null;
           Thread.sleep(1);
       }
       System.out.println(hashSet.remove(new KeyObject()));
   }
}

在这个例子中,我们将new出来的KeyObject对象放进HashSet中。
然后将keyObject置为空。

但是因为类变量hashSet还保留着对keyObject的引用,所以keyObject对象并不会被回收。

注意,最后一行我们加了一个hashSet.remove的代码,来使用类变量hashSet。
为什么要这样做呢?这样做是为了防止JIT对代码进行优化,从而影响我们对内存泄露的分析。

使用JFR和JMC来分析内存泄露

Flight Recorder(JFR)主要用来记录JVM的事件,我们可以从这些事件中分析出内存泄露。

可以通过下面的指令来开启JFR:

java -XX:StartFlightRecording

当然我们也可以使用java神器jcmd来开启JFR:

jcmd pid JFR.dump filename=recording.jfr path-to-gc-roots=true

这里我们使用JMC来图形化分析一下上面的例子。

浅谈JVM之使用JFR解决内存泄露

开启JMC,找到我们的测试程序,打开飞行记录器。

浅谈JVM之使用JFR解决内存泄露

可以看到我们的对象在飞行记录器期间分配了4MB的内存,然后看到整体的内存使用量是稳步上升的。

我们什么时候知道会有内存泄露呢?最简单的肯定就是OutOfMemoryErrors,但是有些很隐蔽的内存泄露会导致内存使用缓步上涨,这时候就需要我们进行细致的分析。

通过分析,我们看到内存使用在稳步上涨,这其实是很可疑的。

接下来我们通过JVM的OldObjectSample事件来分析一下。

OldObjectSample

OldObjectSample就是对生命周期比较长的对象进行取样,我们可以通过研究这些对象,来检查潜在的内存泄露。

浅谈JVM之使用JFR解决内存泄露

这里我们关注一下事件浏览器中的Old Object Sample事件,我们可以在左下方看到事件的详情。

或者你可以使用jfr命令直接将感兴趣的事件解析输出:

jfr print --events OldObjectSample flight_recording_1401comflydeanTestMemoryLeak89268.jfr   > /tmp/jfrevent.log

我们看一个具体的输出Sample:

jdk.OldObjectSample {

  startTime = 19:53:25.607

  allocationTime = 19:50:51.924

  objectAge = 2 m 34 s

  lastKnownHeapUsage = 3.5 MB

  object =  [

    java.lang.Object[200]

  ]

  arrayElements = 200

  root = N/A

  eventThread = "main" (javaThreadId = 1)

  stackTrace = [

    java.util.ArrayList.<init>(int) line: 156

    com.flydean.KeyObject.<init>() line: 11

    com.flydean.TestMemoryLeak.main(String[]) line: 17

  ]

}

lastKnownHeapUsage是heap的使用大小,从日志中我们可以看到这个值是一直在增加的。

allocationTime表示的是这个对象分配的时间。

startTime表示的是这个对象被dump的时间。

object表示的是分配的对象。

stackTrace表示的是这个对象被分配的stack信息。

注意,如果需要展示stackTrace信息,需要开启-XX:StartFlightRecording:settings=profile选项。

从上面的日志我们可以分析得出,main方法中的第17行,也就是 KeyObject keyObject= new KeyObject(); 在不断的创建新的对象。

从而我们可以进行更深层次的分析,最终找到内存泄露的原因。

来源:https://www.cnblogs.com/flydean/p/jvm-diagnostic-memory-leak.html

标签:JVM,JFR,内存泄漏
0
投稿

猜你喜欢

  • C语言new操作的安全性分析

    2021-09-07 05:22:20
  • Struts2中Action中是否需要实现Execute方法

    2021-10-30 06:57:23
  • Android P实现静默安装的方法示例(官方Demo)

    2022-04-05 20:06:13
  • 关于Springboot中JSCH的使用及说明

    2023-11-28 02:32:16
  • c#中使用BackgroundWorker的实现

    2023-05-04 08:51:01
  • Android实现密码明密文切换(小眼睛)

    2023-07-01 15:42:18
  • win7配置adb环境变量的方法步骤

    2022-11-10 16:45:12
  • C#实现为类和函数代码自动添加版权注释信息的方法

    2021-11-30 10:05:21
  • Android实现EditText控件禁止输入内容的方法(附测试demo)

    2021-06-26 08:43:38
  • 关于MyBatis中SqlSessionFactory和SqlSession简解

    2022-02-18 13:57:37
  • C#日期转换函数分享

    2021-06-30 16:48:38
  • C#中+=是什么意思及+=的用法

    2023-07-11 23:25:31
  • java中使用数组进行模拟加密的方法

    2023-11-18 15:37:39
  • Android开发中判断手机是否安装了QQ或者微信

    2022-01-09 08:44:24
  • Android仿人人客户端滑动菜单的侧滑菜单效果

    2021-06-01 08:07:06
  • 基于Android Service 生命周期的详细介绍

    2021-09-11 08:11:42
  • Java项目实战之在线考试系统的实现(系统介绍)

    2022-12-22 11:23:05
  • Java基础之Web服务器与Http详解

    2021-08-13 16:39:42
  • JAVA实现的简单万年历代码

    2021-10-02 21:35:34
  • Android自定义可控制速度的跑马灯

    2023-08-11 14:05:00
  • asp之家 软件编程 m.aspxhome.com