PermGen and Metaspace
PermGen is an abbreviation for Permanent Generation and it’s a special heap space which is separate from the main Java heap where JVM keeps track of metadata of the classes which have been loaded. In Java 8, PermGen has been renamed to Metaspace - with some subtle differences. From our perspective, it is important to note that Metaspace has an unlimited default maximum size. The default size of PermGen memory is 64 MB on 32-bit JVM and 82 MB on the 64-bit version. Of course, these are not the same as the initial sizes. Java 7 and earlier starts with something around 12-21 MB of the initial PermGen space.
JVM |
Default maximum PermGen size (MB) |
Default maximum Metaspace size |
32-bit client JVM |
64 |
unlimited |
32-bit server JVM |
64 |
unlimited |
64-bit JVM |
82 |
unlimited |
It is worthwhile to mention that prior to Java 7, interned Strings used to be kept on the PermGen. That caused some serious problems with the infamous:
java.lang.OutOfMemoryError: PermGen space
Whenever there is a need to resize PermGen/Metaspace, JVM will do it as it does with the standard heap. Resizing those spaces requires a full GC, which is always an expensive operation. It can usually be observed during a startup when a lot of classes are being loaded. Especially if the application has dependencies on many external libraries. If there are a lot of full GCs during the startup, it’s usually because of that. If that case, increasing the initial size can boost the startup performance.
To increase PermGen, we have the following commands:
-XX:PermSize=N
- sets the initial (and minimum size) of the Permanent Generation space.
-XX:MaxPermSize=N
- sets the maximum size of the Permanent Generation space.
In Java 8 and onwards, we can set the initial and maximum size of Metaspace using the following commands:
-XX:MetaspaceSize=N
- sets the initial (and minimum size) of the Metaspace.
-XX:MaxMetaspaceSize=N
- sets the maximum size of the Metaspace.