1. Overview
In this quick tutorial, we're going to get familiar with a few different ways to get the heap size of a running Java application.
2. jcmd
To find the heap and metaspace related info of a running Java application, we can use the jcmd command-line utility:
jcmd GC.heap_info
First, let's find the process id of a particular Java application using the jps command:
$ jps -l
73170 org.jetbrains.idea.maven.server.RemoteMavenServer36
4309 quarkus.jar
12070 sun.tools.jps.Jps
As shown above, the process id for our Quarkus application is 4309. Now that we have the process id, let's see the heap info:
$ jcmd 4309 GC.heap_info
4309:
garbage-first heap total 206848K, used 43061K
region size 1024K, 43 young (44032K), 3 survivors (3072K)
Metaspace used 12983K, capacity 13724K, committed 13824K, reserved 1060864K
class space used 1599K, capacity 1740K, committed 1792K, reserved 1048576K
This app is using the G1 or garbage-first GC algorithm:
- The first line reports the current heap size as 202 MB (206848 K) – also, 42 MB (43061 K) is being used
- G1 regions are 1 MB, there are 43 regions marked as young, and 3 as survivors space
- The current capacity of the metaspace is around 13.5 MB (13724 K). From that 13.5 MB, around 12.5 MB (12983 K) is used. Also, we can have up to 1 GB of metaspace (1048576 K). Moreover, 13842 KB guaranteed to be available for use by the Java virtual machine, also known as committed memory
- The last line shows how much of the metaspace is used to store class information
This output may change depending on the GC algorithm. For instance, if we run the same Quarkus app with ZGC via “-XX:+UnlockExperimentalVMOptions -XX:+UseZGC”:
ZHeap used 28M, capacity 200M, max capacity 1024M
Metaspace used 21031K, capacity 21241K, committed 21504K, reserved 22528K
As shown above, we're using 28 MB of the heap and around 20 MB of metaspace. As of this writing, Intellij IDEA is still using the CMS GC with the following heap info:
par new generation total 613440K, used 114299K
eden space 545344K, 18% used
from space 68096K, 16% used
to space 68096K, 0% used
concurrent mark-sweep generation total 1415616K, used 213479K
Metaspace used 423107K, capacity 439976K, committed 440416K, reserved 1429504K
class space used 55889K, capacity 62488K, committed 62616K, reserved 1048576K
We can spot the classic generational nature of the CMS GC in the heap configuration.
3. jstat
In addition to jcmd, we can use jstat to find out the same information from running applications. For instance, we can use jstat -gc to see heap statistics:
$ jstat -gc 4309
S0C S1C S0U S1U EC EU OC OU MC
0.0 0.0 0.0 0.0 129024.0 5120.0 75776.0 10134.6 20864.0
MU CCSC CCSU YGC YGCT FGC FGCT CGC CGCT GCTGCT
19946.2 2688.0 2355.0 2 0.007 1 0.020 0 0.000 0.027
Each column represents the memory capacity or utilization of a specific memory area:
- S0C — The capacity for the first survivor space
- S1C — The capacity for the second survivor space
- S0U — The used space of the first survivor
- S1U — The used space of the second survivor
- EC — Eden space capacity
- EU — Used space from Eden
- OC — Old generation capacity
- OU — Used space from Old generation
- MC — Metaspace capacity
- MU — Used space from Metaspace
- CCSC — Compressed class space capacity
- CCSU — Used space for compressed classes
- YGC — The number of minor GCs
- YGCT — The time spent for minor GCs
- FGC — The number of full GCs
- FGCT — The time spent for full GCs
- CGC — The number of concurrent GCs
- CGCT — Time spent on concurrent GCs
- GCT — The time spent for all GCs
There are other memory-related options for jstat such as:
- The -gccapacity to report different capacities for different memory regions
- The -gcutil only shows the utilization percentage of each region
- The -gccause is the same as -gcutil but adds the cause of the last GC and possibly current GC events
4. Command-Line Args
If we run a Java application with heap configuration options (for example, -Xms and -Xmx), then there a few other tricks to find the specified values.
For instance, here's how jps reports those values:
$ jps -lv
4309 quarkus.jar -Xms200m -Xmx1g
With this approach, we can only find these static values. So, there is no way to know about, say, the current committed memory.
In addition to jps, a few other tools will report the same thing. For example, the “jcmd <pid> VM.command_line” will also report these details:
$ jcmd 4309 VM.command_line
4309:
VM Arguments:
jvm_args: -Xms200m -Xmx1g
java_command: quarkus.jar
java_class_path (initial): quarkus.jar
Launcher Type: SUN_STANDARD
Also, on most Unix-based systems we can use ps from the procps package:
$ ps -ef | grep quarkus
... java -Xms200m -Xmx1g -jar quarkus.jar
Finally, on Linux, we can use the /proc virtual filesystem and its pid-files:
$ cat /proc/4309/cmdline
java -Xms200m -Xmx1g -jar quarkus.jar
The cmdline file, in a directory named after the Quarkus pid, contains the command-line entry for the application.
5. Conclusion
In this quick tutorial, we saw a few different ways to get the heap size of a running Java application.