博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
使用 Eclipse Memory Analyzer 检测内存泄漏问题
阅读量:4052 次
发布时间:2019-05-25

本文共 1981 字,大约阅读时间需要 6 分钟。

本文是关于在开发Talend RCP 过程中碰到一个内存溢出问题的解决方案。使用的检测软件为Eclipse Memory Analyzer (),是一个开源免费的内存分析工具,目前为eclipse的孵化项目,也是一个eclipse RCP.

Talend一款开源的ETL软件,提供数据集成服务。基于eclipse RCPEMF, GEFJET等技术开发。

 

Talend软件系统中,一个designer editor 是一个mulit page editor, 它的实例是MultiPageTalendEditor,其中包含两页,第一页是一个gef Editor,用来显示GEF图形;第二页是一个JavaEditor,用来显示生成的代码。如下两图所示:

 

 

 

 

    经过其他工具的测试发现JavaEditor如果用在mulit page editor中,关闭后,起实例无法回收,造成内存泄漏,因为Editor是一个非常频繁使用的功能项,所以该问题非常严重。

 

 

要使用Eclipse Memory Analyzer,首先要生成dump文件。

eclipse中配置 Preferences -> Java -> Installed JREs, 定义Jdk1.6来运行Talend.

Talend运行后,执行一下打开editor的操作,在将editor关闭,这样虚拟机中保存了相应的堆栈信息。其实也就是执行一下你认为会产生内存泄漏的操作。

 

运行jdk1.6 bin 目录下的jconsole.exe,运行后如下图所示,选择连接到运行的Talend

  

 

 

 

选择 Mbean -> com.sun.management -> HotSpotDiagnos -> Operations -> dumpHeap

在参数p0中填写保存dump文件的路径,例如f:/my.hprof, 文件的后缀名要为hprof

点击dumpHeap按钮生成dump文件。

 

 

 

 

打开Eclipse Memory Analyzer, 在菜单File 下,选择打开刚才生成的dump文件。如下图:

 

 

 

 

 

在第一行的filter中填写需要检查的类名,这里我要检查的类为TalendJavaEditor。在如下图所示的上下文菜单中选择 List objects -> with incoming references, 查看保存了TalendJavaEditor 实例的引用。

 

 

 

 

 

在如下图所示的上下文菜单中选择 Path To GC Roots -> exclude weak/soft references, 过滤掉弱引用,因为在这里弱引用不是引起问题的关键。

 

 

 

 

 

从下图中,可以看到JavaInfomationProvider中保存了TalendJavaEditor的引用。所以打开JavaInfomationProvider的代码,进行检查。从下图的分析中,大概可以看出这是一个注册了的监听器没有反注册的问题,这也是引起内存泄露的原因中最常见的。

 

 

 

 

public JavaInformationProvider(IEditorPart editor) {

 

       fEditor= editor;

 

       if (fEditor != null) {

 

           fPartListenernew EditorWatcher();

           IWorkbenchWindow windowfEditor.getSite().getWorkbenchWindow();

           window.getPartService().addPartListener(fPartListener);

 

           update();

       }

    }

 

JavaInformationProvider 的构造方法中,保存了TalendJavaEditor的实例,并且注册了监听器。

 

 

 

 

public void partClosed(IWorkbenchPart part) {

           if (part == fEditor) {

              fEditor.getSite().getWorkbenchWindow().getPartService().removePartListener(fPartListener);

              fPartListenernull;

           }

       }

 

 

EditorWatcher的方法partClosed中,当关闭一个designer editor的时候,传进来的partMultiPageTalendEditor,而不是JavaInformationProvider保存的TalendJavaEditor,所以该listener永远不可能被反注册掉,GC也就不可能回收TalendJavaEditor。对于该软件系统来所,这是一个致命的内存溢出问题。

 

 

 

point找到了,解决就不是问题了。

转自:

你可能感兴趣的文章
几个常用的Javascript字符串处理函数 spilt(),join(),substring()和indexof()
查看>>
javascript传参字符串 与引号的嵌套调用
查看>>
swiper插件的的使用
查看>>
layui插件的使用
查看>>
JS牛客网编译环境的使用
查看>>
9、VUE面经
查看>>
关于进制转换的具体实现代码
查看>>
Golang 数据可视化利器 go-echarts ,实际使用
查看>>
mysql 跨机器查询,使用dblink
查看>>
mysql5.6.34 升级到mysql5.7.32
查看>>
dba 常用查询
查看>>
Oracle 异机恢复
查看>>
Oracle 12C DG 搭建(RAC-RAC/RAC-单机)
查看>>
Truncate 表之恢复
查看>>
Oracle DG failover 后恢复
查看>>
mysql 主从同步配置
查看>>
为什么很多程序员都选择跳槽?
查看>>
mongdb介绍
查看>>
mongdb在java中的应用
查看>>
区块链技术让Yotta企业云盘为行政事业服务助力
查看>>