怎么用JDK自带工具进行JVM内存分析

本文涉及的产品
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
云数据库 Tair(兼容Redis),内存型 2GB
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
简介: JVM内存分析工具,如`jps`、`jcmd`、`jstat`、`jstack`和`jmap`,是诊断和优化Java应用的关键工具。`jps`列出Java进程,`jcmd`执行诊断任务,如查看JVM参数和线程堆栈,`jstat`监控内存和GC,`jstack`生成线程堆栈信息,而`jmap`则用于生成堆转储文件。这些工具帮助排查内存泄漏、优化内存配置、性能调优和异常分析。例如,`jmap -dump:file=heapdump.hprof <PID>`生成堆转储文件,之后可以用Eclipse Memory Analyzer (MAT)等工具分析。

进行jvm内存分析可以排查存在和潜在的问题。

通过借助jdk自带的常用工具,可以分析大概可能的问题定位以及确定优化方向。

JVM内存分析有很多好处。

  • 内存泄漏排查:JVM 内存泄漏是指应用程序中的对象占用的内存无法被垃圾回收器释放,导致内存占用持续增长,最终耗尽可用内存。通过内存分析工具,可以检测到哪些对象占用了大量内存且无法被释放,进而定位到可能存在内存泄漏的代码。
  • 内存优化:合理优化 JVM 内存配置可以提高应用程序的性能和稳定性。通过分析应用程序的内存使用情况,可以调整堆内存大小、永久代(如果是旧版 Java)大小、新生代与老年代比例等参数,以减少垃圾回收频率,降低内存占用。
  • 性能调优:内存分析也有助于发现内存中的瓶颈,如频繁的 Full GC(全局垃圾回收)导致的停顿时间过长。通过调整垃圾回收器类型、GC 算法、堆内存大小等参数,可以改善应用程序的性能表现。
  • 异常分析:当应用程序出现内存相关的异常,如 OutOfMemoryError(内存溢出错误)时,通过分析内存使用情况可以找到导致异常的根本原因,例如某个模块或对象占用了过多内存。
  • 容量规划:对于大型应用程序或者需要长时间运行的系统,进行内存分析可以帮助进行容量规划,确保系统具有足够的内存资源支持应用程序的正常运行。

本文将通过一次jvm内存分析过程来说明jpsjcmdjstatjstackjmap 工具的使用方法。

本文使用到的是JDK17版本。

一次jvm内存分析之旅

当需要进行 JVM 内存分析时,结合使用 jpsjcmdjstatjstackjmap 可以提供全面的诊断信息。

下面是一般的步骤:

使用 jps 查看 Java 进程的 PID

bash

代码解读

复制代码

jps -l

这将列出所有 Java 进程的 PID 和主类名。

使用 jcmd 执行一些诊断命令

bash

代码解读

复制代码

jcmd <PID> VM.flags
jcmd <PID> VM.system_properties

这些命令可以显示 JVM 的启动参数和系统属性,帮助了解 JVM 的配置。

使用 jstat 监视 JVM 内存和垃圾回收情况

bash

代码解读

复制代码

jstat -gc <PID> 5000 10

这将持续输出 JVM 的垃圾回收情况,包括各个堆区的使用情况、GC 时间等。

使用 jstack 生成线程堆栈信息

bash

代码解读

复制代码

jstack <PID>

查看线程堆栈信息,以检查是否存在死锁或其他线程相关的问题。

使用 jmap 生成堆转储文件

bash

代码解读

复制代码

jmap -dump:file=heapdump.hprof <PID>

这将生成一个名为 heapdump.hprof 的堆转储文件,可以用于进一步分析内存使用情况,查找内存泄漏等问题。

分析堆转储文件

使用工具如 Eclipse Memory Analyzer (MAT) 或者 VisualVM 来分析生成的堆转储文件,查找内存泄漏、大对象、无用对象等问题。

通过结合使用这些工具,可以全面地了解 JVM 运行时的状态,诊断性能问题,以及解决内存相关的错误。

下面将详细解释这些工具的使用方法。

jps

jps 是 JDK 提供的一个用于列出 Java 虚拟机进程的命令行工具。它通常用于查看当前系统中正在运行的 Java 进程的 PID(进程标识符)以及对应的主类名。

下面是 jps 命令的使用方法:

或者使用ps -ef|grep java 也可以搜索到对应的pid。

bash

代码解读

复制代码

jps [ options ] [ hostid ]

其中,options 是一些可选的参数,hostid 是可选的主机标识符。常用的选项包括:

  • -q:仅显示进程的 PID,不显示对应的主类名。
  • -m:显示传递给主类的参数。
  • -l:显示主类的全限定名,通常用于区分具体的 Java 应用程序。
  • -v:显示传递给 JVM 的参数。

例如,要显示当前系统中所有 Java 进程的 PID 和对应的主类名,可以直接运行 jps 命令:

bash

代码解读

复制代码

jps

如果要仅显示 PID,可以使用 -q 选项:

bash

代码解读

复制代码

jps -q

要显示主类的全限定名,可以使用 -l 选项:

bash

代码解读

复制代码

jps -l

如果要显示传递给主类的参数,可以使用 -m 选项:

bash

代码解读

复制代码

jps -m

如果要显示传递给 JVM 的参数,可以使用 -v 选项:

bash

代码解读

复制代码

jps -v

jcmd

jcmd:jcmd 命令是 Java 8 新增的命令,可以执行多种 JVM 监控和诊断任务。例如,可以使用 jcmd <pid> VM.flags 查看 JVM 启动参数,或者使用 jcmd <pid> Thread.print 打印线程堆栈信息。

下面是 jcmd 命令的基本使用方法:

bash

代码解读

复制代码

jcmd <PID | main class> <command> [options]

其中:

  • <PID | main class>:要操作的 Java 进程的 PID(进程标识符)或者主类名。如果提供了 PID,则直接操作对应的 Java 进程;如果提供了主类名,则 jcmd 会尝试找到匹配的 Java 进程并执行相应的命令。
  • <command>:要执行的诊断命令。
  • [options]:可选的命令参数。

常用的 jcmd 命令包括:

  • help: 显示 jcmd 支持的命令列表,以及每个命令的简要描述。
  • VM.version: 显示 JVM 的版本信息。
  • VM.flags: 显示 JVM 的启动参数。
  • VM.system_properties: 显示 JVM 的系统属性。
  • Thread.print: 打印 Java 进程中所有线程的堆栈信息。
  • GC.run: 执行一次垃圾回收。
  • GC.heap_dump: 生成 Java 堆转储文件(heap dump)。

举例来说,如果要打印指定 Java 进程的线程堆栈信息,可以使用以下命令:

bash

代码解读

复制代码

jcmd <PID> Thread.print

如果要执行一次垃圾回收,可以使用以下命令:

bash

代码解读

复制代码

jcmd <PID> GC.run

如果要生成 Java 堆转储文件,可以使用以下命令:

bash

代码解读

复制代码

jcmd <PID> GC.heap_dump <filename>

jstat

jstat:jstat 命令可以监视 JVM 内存、垃圾回收等情况。例如,可以使用 jstat -gc <pid> 查看垃圾回收统计信息,或者使用 jstat -gcutil <pid> 查看垃圾回收统计信息及内存使用情况。

下面是 jstat 的基本使用方法:

bash

代码解读

复制代码

jstat [ options ] <PID> [ interval [ count ] ]

其中:

  • [ options ]:可选的命令选项,用于指定要监视的数据类型和格式。
  • <PID>:要监视的 Java 进程的 PID(进程标识符)。
  • [ interval ]:可选参数,指定输出统计信息的时间间隔(以毫秒为单位)。如果省略,则默认为每秒一次。
  • [ count ]:可选参数,指定输出统计信息的次数。如果省略,则默认为无限次输出。

常用的 jstat 命令选项包括:

  • -class: 显示类加载、卸载信息以及类加载器的状态。
  • -gc: 显示垃圾回收相关的信息,包括各个代的使用情况、GC 时间等。
  • -compiler: 显示即时编译器(JIT)的编译统计信息。
  • -gccapacity: 显示各个堆区的容量及使用情况。
  • -gcutil: 显示各个堆区的使用情况,以百分比表示。
  • -gccause: 显示导致最近一次 GC 的原因。
  • -printcompilation: 打印方法的即时编译(JIT)信息。

举例来说,要查看 Java 进程的类加载情况,可以使用以下命令:

bash

代码解读

复制代码

jstat -class <PID>

如果想要每隔 5 秒输出一次类加载信息,共输出 10 次,可以使用以下命令:

bash

代码解读

复制代码

jstat -class <PID> 5000 10

jstat只能查看当前的gc信息,查看gc日志更适合线上环境的做法是在启动JVM时加上-XX:+PrintGCDetails -Xloggc:/path/to/gc.log(JDK1.8以下)或者-Xlog:gc*:file=/path/to/gc.log(JDK9+)参数,将生成的gc日志文件放到GCViewer、GCeasy(需注册)进行分析。

jstack

jstack:jstack 命令用于生成 Java 线程转储快照,可以用于分析线程状态、死锁等问题。例如,可以使用 jstack <pid> 打印线程堆栈信息,或者使用 jstack -l <pid> 打印线程堆栈信息及锁信息。

下面是 jstack 命令的基本使用方法:

bash

代码解读

复制代码

jstack [ options ] <PID>

其中:

  • [ options ]:可选的命令选项,用于指定输出的格式等。
  • <PID>:要生成线程堆栈信息的 Java 进程的 PID(进程标识符)。

常用的 jstack 命令选项包括:

  • -l: 长列表格式,显示关于锁的附加信息,如拥有者和等待队列。
  • -F: 当正常输出的 jstack 命令不起作用时,强制生成线程堆栈信息。这在 Java 进程没有响应时可能会很有用,但可能会导致进程暂停一段时间。
  • -m: 显示 Java 和本地方法的堆栈跟踪,而不仅仅是 Java 堆栈跟踪。
  • -h: 显示帮助信息。

举例来说,要生成指定 Java 进程的线程堆栈信息,可以使用以下命令:

bash

代码解读

复制代码

jstack <PID>

如果想要输出长列表格式的线程堆栈信息,可以使用 -l 选项:

bash

代码解读

复制代码

jstack -l <PID>

如果 Java 进程没有响应,可以使用 -F 选项强制生成线程堆栈信息:

bash

代码解读

复制代码

jstack -F <PID>

jmap

异常没有发生定位异常代码,需要通过jmap生成dump文件。

然后将其导入到 MAT 中进行分析。以下是生成堆转储文件的步骤:

  • 确定 Java 进程 ID:首先,需要确定正在运行的 Java 进程的进程 ID(PID)。可以使用 jps 命令查看正在运行的 Java 进程及其 PID。
  • 生成堆转储文件:使用 jmap 命令生成堆转储文件。命令格式如下:

shell

代码解读

复制代码

jmap -dump:file=<文件路径> <PID>

例如,要生成名为 heapdump.hprof 的堆转储文件,可以执行以下命令:

shell

代码解读

复制代码

jmap -dump:file=heapdump.hprof <PID>

这将在当前工作目录下生成一个名为 heapdump.hprof 的堆转储文件。

  • 导入堆转储文件到 MAT:将生成的堆转储文件导入到 MAT 中进行分析。打开 MAT,然后选择 File -> Open Heap Dump,然后选择生成的堆转储文件。
  • 执行内存分析:一旦堆转储文件被导入到 MAT 中,就可以执行内存分析,按照前面提到的步骤来查找内存问题。

通过这些步骤可以手动生成堆转储文件并使用 MAT 进行分析,即使没有在 OutOfMemoryError 发生时自动生成堆转储文件也可以找到问题所在。

更适合线上环境的做法是在启动JVM时加上-XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=heapdump.hprof参数,这样当发生OutOfMemoryError时,JVM会自动生成堆转储文件。


转载来源:https://juejin.cn/post/7359374598083641354

相关文章
|
3月前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
544 1
|
3月前
|
监控 Oracle Java
JDK 21中的分代ZGC:一场内存管理的革命
JDK 21引入了分代ZGC,为Java应用程序的内存管理带来了革命性的进步。分代ZGC通过将堆内存划分为年轻代和老年代,采用并发处理和染色指针技术,实现了高吞吐量、低延迟和更好的可扩展性。这一特性显著提升了系统的性能和稳定性。
280 51
|
2月前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
3月前
|
Arthas Prometheus 监控
监控堆外使用JVM工具
监控堆外使用JVM工具
56 7
|
3月前
|
JavaScript
如何使用内存快照分析工具来分析Node.js应用的内存问题?
需要注意的是,不同的内存快照分析工具可能具有不同的功能和操作方式,在使用时需要根据具体工具的说明和特点进行灵活运用。
67 3
|
3月前
|
Oracle 安全 Java
深入理解Java生态:JDK与JVM的区分与协作
Java作为一种广泛使用的编程语言,其生态中有两个核心组件:JDK(Java Development Kit)和JVM(Java Virtual Machine)。本文将深入探讨这两个组件的区别、联系以及它们在Java开发和运行中的作用。
139 1
|
3月前
|
Java 编译器 API
深入解析:JDK与JVM的区别及联系
在Java开发和运行环境中,JDK(Java Development Kit)和JVM(Java Virtual Machine)是两个核心概念,它们在Java程序的开发、编译和运行过程中扮演着不同的角色。本文将深入解析JDK与JVM的区别及其内在联系,为Java开发者提供清晰的技术干货。
51 1
|
3月前
|
Java
JVM内存参数
-Xmx[]:堆空间最大内存 -Xms[]:堆空间最小内存,一般设置成跟堆空间最大内存一样的 -Xmn[]:新生代的最大内存 -xx[use 垃圾回收器名称]:指定垃圾回收器 -xss:设置单个线程栈大小 一般设堆空间为最大可用物理地址的百分之80
|
3月前
|
Java
JVM运行时数据区(内存结构)
1)虚拟机栈:每次调用方法都会在虚拟机栈中产生一个栈帧,每个栈帧中都有方法的参数、局部变量、方法出口等信息,方法执行完毕后释放栈帧 (2)本地方法栈:为native修饰的本地方法提供的空间,在HotSpot中与虚拟机合二为一 (3)程序计数器:保存指令执行的地址,方便线程切回后能继续执行代码
34 3
|
3月前
|
存储 缓存 监控
Elasticsearch集群JVM调优堆外内存
Elasticsearch集群JVM调优堆外内存
67 1