Java - lang.OutOfMemoryError - JAVA HEAP SPACE
Java - lang.OutOfMemoryError - JAVA HEAP SPACE
Java - lang.OutOfMemoryError - JAVA HEAP SPACE
…\NWDS\NWCEIDE07P_8-80002366\eclipse\eclipse.ini
If you do not explicitly set the sizes, platform-specific defaults will be used.
The java.lang.OutOfMemoryError: Java heap space error will be triggered when the
application attempts to add more data into the heap space area, but there is not enough
room for it.
There most common reason for the java.lang.OutOfMemoryError: Java heap spaceerror is
simple – you try to fit an XXL application into an S-sized Java heap space. That is – the
application just requires more Java heap space than available to it to operate normally. Other
causes for this OutOfMemoryError message are more complex and are caused by a
programming error:
• Spikes in usage/data volume. The application was designed to handle a certain amount of
users or a certain amount of data. When the number of users or the volume of data
suddenly spikes and crosses that expected threshold, the operation which functioned
normally before the spike ceases to operate and triggers the java.lang.OutOfMemoryError:
Java heap space error.
• Memory leaks. A particular type of programming error will lead your application to
constantly consume more memory. Every time the leaking functionality of the application is
used it leaves some objects behind into the Java heap space. Over time the leaked objects
consume all of the available Java heap space and trigger the already
familiar java.lang.OutOfMemoryError: Java heap space error.
The first example is truly simple – the following Java code tries to allocate an array of 2M
integers. When you compile it and launch with 12MB of Java heap space (java -Xmx12m OOM),
it fails with the java.lang.OutOfMemoryError: Java heap spacemessage. With 13MB Java heap
space the program runs just fine.
class OOM {
static final int SIZE=2*1024*1024;
public static void main(String[] a) {
int[] i = new int[SIZE];
}
}
It is fairly easy to construct a Java program that satisfies the definition of a memory leak:
class KeylessEntry {
Key(Integer id) {
this.id = id;
}
When you execute the above code above you might expect it to run forever without any
problems, assuming that the naive caching solution only expands the underlying Map to 10,000
elements, as beyond that all the keys will already be present in the HashMap. However, in
reality the elements will keep being added as the Key class does not contain a
proper equals() implementation next to its hashCode().
As a result, over time, with the leaking code constantly used, the “cached” results end up
consuming a lot of Java heap space. And when the leaked memory fills all of the available
memory in the heap region and Garbage Collection is not able to clean it,
the java.lang.OutOfMemoryError:Java heap space is thrown.
The solution would be easy – add the implementation for the equals() method similar to the
one below and you will be good to go. But before you manage to find the cause, you will
definitely have lose some precious brain cells.
@Override
public boolean equals(Object o) {
boolean response = false;
if (o instanceof Key) {
response = (((Key)o).id).equals(this.id);
}
return response;
}
In some cases, the amount of heap you have allocated to your JVM is just not enough to
accommodate the needs of your applications running on that JVM. In that case, you should just
allocate more heap – see at the end of this chapter for how to achieve that.
In many cases however, providing more Java heap space will not solve the problem. For
example, if your application contains a memory leak, adding more heap will just postpone
the java.lang.OutOfMemoryError: Java heap space error. Additionally, increasing the amount of
Java heap space also tends to increase the length of GC pauses affecting your
application’s throughput or latency.
If you wish to solve the underlying problem with the Java heap space instead of masking the
symptoms, you need to figure out which part of your code is responsible for allocating the most
memory. In other words, you need to answer these questions:
1. Which objects occupy large portions of heap
2. where these objects are being allocated in source code
At this point, make sure to clear a couple of days in your calendar (or – see an automated way
below the bullet list). Here is a rough process outline that will help you answer the above
questions:
• Get security clearance in order to perform a heap dump from your JVM. “Dumps” are
basically snapshots of heap contents that you can analyze. These snapshot can thus contain
confidential information, such as passwords, credit card numbers etc, so acquiring such a
dump might not even be possible for security reasons.
• Get the dump at the right moment. Be prepared to get a few dumps, as when taken at a
wrong time, heap dumps contain a significant amount of noise and can be practically
useless. On the other hand, every heap dump “freezes” the JVM entirely, so don’t take too
many of them or your end users start facing performance issues.
• Find a machine that can load the dump. When your JVM-to-troubleshoot uses for example
8GB of heap, you need a machine with more than 8GB to be able to analyze heap contents.
Fire up dump analysis software (we recommend Eclipse MAT, but there are also equally
good alternatives available).
• Detect the paths to GC roots of the biggest consumers of heap. We have covered this
activity in a separate post here. It is especially tough for beginners, but the practice will
make you understand the structure and navigation mechanics.
• Next, you need to figure out where in your source code the potentially hazardous large
amount of objects is being allocated. If you have good knowledge of your application’s
source code you’ll be able to do this in a couple searches.
Equipped with this information you can zoom in to the underlying root cause and make sure the
data structures are trimmed down to the levels where they would fit nicely into your memory
pools.
However, when your conclusion from memory analysis or from reading the Plumbr report are
that memory use is legal and there is nothing to change in the source code, you need to allow
your JVM more Java heap space to run properly. In this case, alter your JVM launch
configuration and add (or increase the value if present) the following:
-Xmx1024m
The above configuration would give the application 1024MB of Java heap space. You can use g
or G for GB, m or M for MB, k or K for KB. For example all of the following are equivalent to
saying that the maximum Java heap space is 1GB:
The main difference from a user perspective - Metaspace by default auto increases its size (up
to what the underlying OS provides), while PermGen always has a fixed maximum size. You can
set a fixed maximum for Metaspace with JVM parameters, but you cannot make PermGen auto
increase.
To a large degree it is just a change of name. Back when PermGen was introduced, there was
no Java EE or dynamic class(un)loading, so once a class was loaded it was stuck in memory until
the JVM shut down - thus Permanent Generation. Nowadays classes may be loaded and
unloaded during the lifespan of the JVM, so Metaspace makes more sense for the area where
the metadata is kept.
Both of them contain the java.lang.Class instances and both of them suffer from ClassLoader
leaks. Only difference is that with Metaspace default settings, it takes longer until you notice
the symptoms (since it auto increases as much as it can), i.e. you just push the problem further
away without solving it. OTOH I imagine the effect of running out of OS memory can be more
severe than just running out of JVM PermGen, so I'm not sure it is much of an improvement.