I am having some memory issues with my application and need help understanding these statistics.
Unix 'top' shows these stats for my process-
VSZ: 37.4g
RSS: 20.0g
So, this means 20g is currently swapped in for the process and in use.
However, when I print stats from within my application using Runtime class, I get this:
Runtime.totalMemory() : 9.8G
Runtime.freeMemory() : 3.6G
Runtime.maxMemory() : 14.3G
Why doesn't [Runtime.totalMemory() - Runtime.freeMemory()] match RSS? This is the memory currently in use by the process. There is a huge difference between the two numbers.
Also, does the runtime give back the unused memory (Runtime.freeMemory()) back to OS for use by other processes?
Note that my applications are running in a peer to peer GemFire caching system set up with shared and replicated caches. I need to optimize the application to reduce memory footprint.
Runtime.totalMemory shows currently available memory. Java allocates memory lazily.
No. If Java allocated memory (totalMemory) it is in java process now.
As already mentioned Java uses memory besides heap. Also Gemfire uses off heap memory (check this). Try to look at them in VisualVM-Buffer Monitor.
What is your infrastracture (OS, VMs)?
If you can't use standard tools, you probably should write own serviceability agent using JMX (for example)
UPDATE
Ok
If you still ask, I would provide more details. What are java memory usage?
To see default values, you can run:
To see which options for your particular process, you can run:
For some of them yes. For some of them no.
For direct memory you could try sun.misc.SharedSecrets.getJavaNioAccess().getDirectBufferPool().getMemoryUsed().
With cmd, for example:
But it depends on running settings (NativeMemoryTracking). You could read how to enable Native Memory tracking here.
In linux you also could use:
In conclusion this is probably no matter, because your task:
And you can not impact on native memory usage. I suppose look at jmap util, which can show you class histogram. You should check what has big size in GemFire and review those objects, probably you store in cache data, which are not should be there. I mean in my practice for optimizing cache I review object and fields and see what fields are really frequent, what are not. Another approach is check your serialization mechanism and object layouts.
As I mentioned you can use serviceability agents:
You need tools.jar from JDK in dependencies to run this. It could print you class histogram too, but sometimes works, when jmap does not. Also, when you should wait a lot of time, while histogram is calculated, you could use VM.getSystemDictionary() and find only your classes. And, it would be usefull, if you can not enable NMT, because of overhead.