浅谈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来图形化分析一下上面的例子。
开启JMC,找到我们的测试程序,打开飞行记录器。
可以看到我们的对象在飞行记录器期间分配了4MB的内存,然后看到整体的内存使用量是稳步上升的。
我们什么时候知道会有内存泄露呢?最简单的肯定就是OutOfMemoryErrors,但是有些很隐蔽的内存泄露会导致内存使用缓步上涨,这时候就需要我们进行细致的分析。
通过分析,我们看到内存使用在稳步上涨,这其实是很可疑的。
接下来我们通过JVM的OldObjectSample事件来分析一下。
OldObjectSample
OldObjectSample就是对生命周期比较长的对象进行取样,我们可以通过研究这些对象,来检查潜在的内存泄露。
这里我们关注一下事件浏览器中的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


猜你喜欢
- 前言在项目中,如果我们要遵循分层领域模型规约: 话,肯定避免不了在DTO、VO、BO、AO、VO、Query等实体的转换,我们通常有几种做法
- 在使用STL容器(比如map、list、vector等)的时候,是用放一个对象还是放一个对象指针,即是用vector<int>还
- socket使用getInputStream()阻塞今天用socket进行编程练习时,发现程序到了getInputStream()这里就进行
- 一、项目简述功能包括:用户分为宠物,医生,管理员,宠物主人可进行注册选择医生挂号,选择日期,选择号源,医生可进行宠物接诊,管理员可对宠物,医
- 步骤:1、创建一个项目,该项目主要用来设计用户控件。2、创建一个用户控件窗体,用来设计用户控件。3、向用户控件窗体中添加一个按钮(butto
- 时钟的使用1、声明VOID CALLBACK playproc( HWND hwnd, &nb
- Contact联系人对Mms来说是十分重要的,因为每一个对话的收信人都是一个联系人,新建信息时可以输入联系人的任何信息,比如号码或名字,Mm
- SpringBoot找不到映射文件org.apache.ibatis.binding.BindingException: Invalid b
- 由于最近想要阅读下JDK1.8 中HashMap的具体实现,但是由于HashMap的实现中用到了红黑树,所以我觉得有必要先复习下红黑树的相关
- Filter过滤器中访问getSession()要进行转化public void doFilter(ServletRequest reque
- 不依赖任何外界包,maven如何生成可以执行的jar?pom中不包含任何引用的情况下,只需要在pom中添加 maven-jar-plugin
- 一、前言我们在日常开发中,避不开的就是参数校验,有人说前端不是会在表单中进行校验的吗?在后端中,我们可以直接不管前端怎么样判断过滤,我们后端
- 前言在SpringIOC中,我们熟知的BeanScope有单例(singleton)、原型(prototype), Bean的Scope影响
- 学会了Paint,Canvas的基本用法之后,我们就可以动手开始实践了,先写个简单的图片加载进度条看看。按照惯例,先看效果图,再决定要不要往
- 1.问题分析我们在开发中经常遇到多个实体类有共同的属性字段,例如在用户注册时需要设置创建时间、创建人、修改时间、修改人等字段,在用户编辑信息
- Java计算一段程序的运行时间介绍了两种方法,一种是毫秒级别的计算,另一种是更精确的纳秒级别的计算。毫秒级别计算时间  
- spring data jpa使用自定义repository实现类spring data jpa中使用JpaRepository等接口定义r
- 出错信息:Unknown error (0xffffffff)at System.Diagnostics.Process.StartWith
- 在拿到一个 Stream 如何优雅将这个 Stream 保存到代码最优雅的方法应该是通过 CopyTo 或 CopyToAsync 的方法u
- 本文实例为大家分享了C#使用NPOI将excel导入到list的具体代码,供大家参考,具体内容如下这个是确定是实体类接收/// <su