当前位置:网站首页 > 编程语言 > 正文

jvisualvm分析dump大对象(jvm dump分析)



本文讲解了 JVM 的内存划分和分配策略,并以截图和脚本展示常用可视化和命令行工具的使用方法,完整演示了 JVM 优化、内存泄露排查、gc.log 分析方法等。

1.1 内存划分

1.1.1 堆(Heap)

存放 new MyClass() 的对象,是GC的主要区域,

-Xms / -Xmx 分别是堆的初始容量、最大可扩展容量,建议初始值设置为最大值,以免反复扩展或缩减的开销;

新生代(Young Generation):又划分为 Eden(伊甸园,新生区), Survivor#0(幸存区S0), Survivor#1(幸存区S1)

老年代(Tenured Generation)

-XX:NewRatio 是“老年代 / 新生代”的比例,默认值为 2;

-XX:SurvivorRatio 是指 Eden/Survivor#0 的比例,而Survivor#0 与 Survivor#1 容量相同;

1.1.2 永久代(Permanent Generation)

存放类信息、常量、大对象(比如 new byte[n]对象);

-XX:PermSize / -XX:MaxPermSize 为永久代的初始、最大容量,建议初始容量指定为最大容量;

1.1.3 类元数据(Metaspace)

JDK8 中,永久代被完全的移除了(相关参数 -XX:PermSize / -XX:MaxPermSize 被忽略)。改用 Metaspace,相关参数如下:

-XX:MetaspaceSize

-XX:MaxMetaspaceSize: 最大容量,默认没有限制(机器内存);

-XX:CompressedClassSpaceSize

1.1.4 实例解析

选项 -Xms300M -Xmx300M -XX:NewRatio=2 -XX:SurvivorRatio=8 -XX:PermSize=100M -XX:MaxPermSize=100M 含义为:

永久代固定尺寸为 100M;

整个堆固定尺寸为 300M,其中“老年代 / 新生代”为-XX:NewRatio=2,所以老年代为 200M,新生代为 100M;

新生代总共 100M,其中“Eden / Survivor0”为-XX:SurvivorRatio=8,所以 Eden 为 80M,Survivor0=Survivor1=10M。

官方资料:

http://www.oracle.com/technetwork/articles/java/vmoptions-jsp-140102.html

1.2 JVM内存分配策略

1.2.1. 对象优先分配在 Eden

首先尝试在 Eden 分配,若 Eden 空间不足,就发起 YoungGC(简称YGC);

如果幸存对象(被根对象直接或间接引用)在 SurvivorTo 能放下,则存活对象全部移入 SurvivorTo;

如果幸存对象在 SurvivorTo 存放不下,则存活对象全部移入老年代;

如果此时老年代空间不足,则发起一次 FullGC(简称FYC,整个系统停顿,老年代也参与回收);

如果 FullGC 时空间仍然周转不过来,则报 OutOfMemoryError 并导致进程结束;

如果YGC/FGC能周转过来,则新对象分配在 Eden 中;

1.2.2. 大对象直接分配在老年代

所谓的大对象是指,需要大量连续内存的 Java 对象,尤其是长字符串或数组,比如 byte[n] 对象;

可以指定多大才算大对象(只对 Serial/ParNew 有效);

1.2.3. 长期存活对象移入老年代

经历 n 次 YGC 仍然存活的对象,下次 YGC 时将被移入老年代;

可以设置该数值:

-XX:MaxTenuringThreshold=15(默认)

1.2.4. 永久代满了也会导致 FullGC

老年代、永久代的垃圾收集是捆绑在一起的,因此无论两者谁满了,都会触发两者的FullGC。

1.2.5. 动态对象年龄判定

如果 Survivor 相同年龄对象占用空间达到一半,则大于等于该年龄的对象都移入老年代;

1.2.6. 冒险模式

JDK 6u24 之后,总是开启冒险模式(先尝试 YGC,以免 FGC 频繁);

每次 YGC 之前,如果老年代最大连续空间,大于新生代所有对象空间之和,则 YGC 肯定成功,否则:

如果老年代最大连续空间,大于历次晋升老年代的平均值,则先冒险尝试 YGC;

如果老年代最大连续空间,小于历次晋升老年代的平均值,则直接 FullGC;

2.1 优化目标

2.1.1 尽量减少 YoungGC
2.1.2 尽量减少 FullGC

一天最多 FullGC 一次,最好在系统空闲期(如深夜);

2.2 优化方法

2.2.1 代码角度

缩短对象生命期,尤其是大对象

2.2.2 JVM参数角度

优化JVM参数以减少YGC/FGC次数,可替换收集器

这两项功能必须开启,下面的可视化工具 VisualVM 要用到。

3.1 设置系统环境变量

重新登录后生效。选项 java.rmi.server.hostname 的作用:服务器把该值传给 VisualVM,VisualVM 使用该地址查找 RMI 服务,所以必须是客户可以访问的 RMI 服务器的外网地址。

3.2 开启 JMX(指定端口 1090)

需要注意的是,如果服务端 JMX 开启了修改和控制权限,此时如果不验证监控客户端的身份,那么所有用户都可以修改和控制 Tomcat 服务,所以重要的服务器应该开启用户名和密码验证。

3.2.1 准备用户验证文件

jmxremote.password 用于设置各个【用户名|密码】

jmxremote.access 用于设置各个【用户名|权限】

3.2.2 修改用户验证文件权限

安全起见,JMX 限制其他用户不可读这两个用户验证文件。

默认情况下,Windows/Linux 下分别会报如下错误:

错误: 必须限制口令文件读取访问权限:

%CATALINA_BASE%confjmxremote.password

Error: Password file read access must be restricted:

$CATALINA_BASE/conf/jmxremote.password

Windows下可按如下操作修改文件权限:

右键单击文件 jmxremote.password,弹出菜单中选“属性”,再点“安全”/“高级”/“更改权限”/“包括可从该对象的父项继承的权限”(弹出窗口中选“删除”以删除所有访问权限);

再选“添加”/“高级”/“立即查找”,选中你的用户(如 WKF-PC),点“确定”;

权限项目窗口中勾选“完全控制”,点“确定”。

Linux 下则更简单:

3.2.3 修改 Tomcat 启动时 JVM 选项

如果测试服务器无需开启用户验证,只需修改下面参数 authenticate=false, 并去掉 password.file 和 access.file 两个参数。

重启 Tomcat 生效。

3.3 开启 jstatd agent(默认端口 1099)

执行 $CATALINA_HOME/bin/jstatd.sh 后生效。

3.4 配置防火墙(放行端口 1090/1099)

4.1 JConsole4 可视化工具

早期的 Java 故障和监控工具,现在可以由强大的 VisualVM 代替。

4.2 VisualVM

4.2.1 文档

http://m.blog.csdn.net/article/details?id=

http://docs.oracle.com/javase/8/docs/technotes/guides/visualvm/index.html

4.2.2 下载与安装

可以使用 JDK 自带的 %JAVA_HOME%binjvisualvm.exe;

也可以使用最新版本,下载方法如下:

下载页:https://visualvm.github.io/download.html

主程序:

https://github.com/visualvm/visualvm.src/releases/download/1.3.9/visualvm_139.zip

汉化包:

https://github.com/visualvm/visualvm.src/releases/download/1.3.9/visualvm_139-ml.zip

IDE插件:https://visualvm.github.io/idesupport.html

主程序与汉化包解压到同一目录,最终主程序为 binvisualvm.exe;

IDE插件支持 Eclipse/IntelliJ IDEA,可以随着 IDE 启动。

配置 VisualVM 插件中心(默认的配置中心已失效):

菜单:工具 / 插件 / “设置”、“编辑”,修改为:

https://visualvm.github.io/uc/release139/updates.xml.gz

更多可用地址列表:https://visualvm.github.io/pluginscenters.html

4.2.3 安装 VisualGC 插件

http://www.oracle.com/technetwork/java/visualgc-136680.html

VisualVM 中,点击菜单“工具”/“插件”,如下图安装:

有些插件不能成功安装,可下载 nbm 包再手工安装:

https://visualvm.github.io/plugins.html

http://visualvm.java.net/pluginscenters.html

或者在插件设置里,修改或新增插件源(pluginscenters.html里点击后能找到),再安装。

http://bits.netbeans.org/VisualVM/uc/release138/updates.xml

http://bits.netbeans.org/VisualVM/uc/8u40/updates.xml

几款有用的插件:

BTrace Workbench(只适用于本地应用): 用于不停止进程的情况下添加代码跟踪,入口在应用的右键菜单里;

KillApplication(只适用于本地应用): 用于杀掉进程,入口在应用的右键菜单里;

VisualVM-MBeans(本地和远程通用):类似于 JConsole 展示应用的 MBean,包括值、操作、通知等;

Tracer(本地和远程通用): 以时间曲线图展示多种指标,包括CPU/GC/Heap/PermGen/Classes/Threads等。

4.2.4 前提条件

对于本地应用,VisualVM 所有功能直接支持它;

对于远程应用,VisualGC 标签页要求服务端开启 jstatd agent,其他标签页要求开启 JMX,否则会显示错误“不受此 JVM 支持”。

对于远程应用,需要注意的是,服务端 jstatd/JMX 重启后,VisualVM 必须重启或者重建 JMX 连接,否则服务端调整在 VisualVM 中不生效。

4.2.5 添加远程主机 / JMX连接

(1)添加“远程主机”,指定远程服务器的 IP 和 jstatd 端口:

(2)添加“JMX 连接”,指定远程应用的 JMX 端口、用户名和密码:

如遇连接失败,请检查 JMX_HOSTNAME 没有生效的原因。

4.2.6 监控远程应用

(1)双击左侧的“JMX 连接”(注意小图标底部有 JMX 字样),切换至“概述”标签页,可看到概述和 JVM 参数信息:

(2)切换至“监视”标签页,可看到 CPU、Heap、Class加载、线程等时间曲线图:

(3)切换至“线程”标签页,可看到各线程 CPU 耗时统计:

(4)切换至“抽样器”标签页,可看到热点方法耗时、主要对象占用空间统计(可用于定位内存泄露):

(5)切换至“VisualGC”标签页,可看到各种内存变化曲线、各GC时间点(可用于JVM参数调优):

4.2.7 JVM 优化实战

(1)优化前,没有明确指定各内存大小,使用 Java 默认内存大小,相当于指定为:

可见 YoungGC 很频繁:

观察图形可得出以下优化方案:

机器内存为 1G,本应用为系统唯一大应用,分给它 512M,另外 512M 预留给系统和其他应用;

永久代只需 30M,可指定初始和最大值为 64M,避免反复伸缩的开销;

整个堆(新生代+老年代)目前内存为 240M,可增加至 384M,指定初始和最大值都为 384M,避免反复伸缩的开销;

YoungGC 过于频繁,原因是 Eden 区过小。“老年代/新生代”目前比例为 -XX:NewRatio=2,看图形可知,老年代中长寿对象并不多,可缩减老年代让给新生代,所以调整 -XX:NewRatio=1;

“Eden / Survivor0”比值目前为 -XX:SurvivorRatio=8,暂不调整。

(2)优化后,指定 JVM 选项为:

可见 YoungGC 大幅减少:

(3)优化前后对比:

优化前:YoungGC 102次,总耗时 965ms, FullGC 5次,总耗时 240ms,而且应用稳定后 YoungGC 仍然反复发生;

优化后:YoungGC 3次,总耗时 26ms, FullGC 1次,总耗时 4ms,而且应用稳定后 YoungGC 长期未再发生;

优化效果非常明显。

5.1 jps(查看Java进程)

5.2 jmap/jhat(快照的生成与查看)

5.3 jstack

5.4 jcmd

6 开发监控系统

假设有个需求(这里只是为了演示,现实中不会这么简单粗暴,现实中还需要限制在凌晨才触发):Heap 使用率高于 60%,就要求强制 FullGC。

6.1 Shell 脚本

6.2 Java 代码

代码中使用的“java.lang:type=Memory”、“HeapMemoryUsage”可以借助 VisualVM/MBeans插件的 Attribtes/Operations 标签页中找到,如下图:

7 内存泄露插件

7.1 内存泄露现象

老年代越来越大,GC越来越频繁、执行时间越来越长,而且GC后内存未释放。

7.2 生成快照文件(hprof Heap信息文件)

使用Java的jmap命令来生成;

通过JMX的MBean用Java代码生成;

7.3 分析dump文件

最佳方案是使用 Eclipse MAT 插件;

其他候选方案:Java 自带工具 Visual VM / jhat、IBM HeapAnalyzer;

7.4 分析内存泄漏

可看到可疑的内存泄露对象,看到占用空间大的对象及其调用关系。

8.1 开启gc日志文件

8.2 日志文件格式解析

[停顿类型

  [新生代: GC前本区域已用容量 -> GC后本区域已用容量 (本区域总容量), 本区域GC耗时]

  [老年代: GC前本区域已用容量 -> GC后本区域已用容量 (本区域总容量), 本区域GC耗时]

    GC前Heap已用容量 -> GC后Heap已用容量 (Heap总容量)

  [永久代: GC前本区域已用容量 -> GC后本区域已用容量 (本区域总容量), 本区域GC耗时]

  [Times: 用户耗时=xx 系统耗时=yy, 实际耗时=zz secs]

==== server 模式的 GC+FullGC ====

新生代:PSYoungGen=Parallel Scavenge

老年代:ParOldGen=Parallel Old

永久代:PSPermGen

================== client 模式的 GC+FullGC ===================

新生代:DefNew=Serial(Default New Generation)

老年代:Tenured

永久代:Perm

===================================================

8.3 gc.log 解析工具

8.3.1 在线分析

http://gceasy.io/

8.3.2 ga456(IBM)

https://www.ibm.com/developerworks/community/alphaworks/tech/pmat

ftp://public.dhe.ibm.com/software/websphere/appserv/support/tools/pmat/ga456.jar

8.3.3 HPjmeter

https://h20392.www2.hpe.com/portal/swdepot/displayProductInfo.do?productNumber=HPJMETERSW

需要注册帐号并登录,最终下载页中选择“Use Standard Download”,Windows上可下载这两个:

MS Windows XP/Vista/7 HPjmeter 4.4.00.00 Console - Oct 2014 (Z7550-01611_hpjmeter_console_4.4.00.00_windows_setup.exe)

HPjmeter 4.4.00.00 Console jar file zip format - Feb 2015 (Z7550-63266_hpjmeter_4.4.00.00.zip)

8.3.4 GCViewer

http://www.tagtraum.com/gcviewer.html

http://www.tagtraum.com/gcviewer-download.html

http://www.tagtraum.com/download/gcviewer-1.29-bin.zip

8.3.5 IBM GCMV(Eclipse 插件)

GCMV=Garbage Collection and Memory Visualizer

插件中心搜索 “GCMV” 来安装,然后在菜单中打开: Window / Perspective / Open Perspective / GCMV

8.3.6 gchisto(VisualVM 插件)

https://gchisto.dev.java.net/

http://java.net/projects/gchisto

http://pietrowski.info/wp-content/uploads/2009/06/GCHisto.tgz

8.3.7 其他更多

https://code.google.com/archive/p/gclogviewer/

https://code.google.com/archive/p/verbosegcanalyzer

http://fasterj.com/tools/gcloganalysers.shtml

http://techblog.netflix.com/2013/05/garbage-collection-visualization.html

作者:王克锋

出处:https://kefeng.wang/2016/11/22/java-jvm/

到此这篇jvisualvm分析dump大对象(jvm dump分析)的文章就介绍到这了,更多相关内容请继续浏览下面的相关推荐文章,希望大家都能在编程的领域有一番成就!

版权声明


相关文章:

  • 苹果怎么查看密码库的文件(苹果手机怎么看密码库)2024-12-29 20:45:09
  • linux 怎么安装yum(linux怎么安装yum源)2024-12-29 20:45:09
  • 国内github平替(github中国区排行榜)2024-12-29 20:45:09
  • 天国拯救战斗系统怎么格挡(天国拯救怎么格挡反击)2024-12-29 20:45:09
  • 流量录制与回放(流量录制回放是什么意思)2024-12-29 20:45:09
  • issa英文名(issa英文名含义)2024-12-29 20:45:09
  • 二级域名解析882gg(二级域名解析网站)2024-12-29 20:45:09
  • 速排卵药是什么(排卵药是什么药)2024-12-29 20:45:09
  • 苹果电脑装双系统开机怎么切换(苹果电脑装完双系统怎么切换)2024-12-29 20:45:09
  • max3232原理图(max3232中文资料)2024-12-29 20:45:09
  • 全屏图片