- Диагностика утечек памяти в Java
- Инструменты
- Анализ использования памяти
- Выявление утечки
- Устранение утечки
- How to Detect Memory Leaks in Java: Causes, Types, & Tools
- What Causes Memory Leaks in Java
- Types of Memory Leaks in Java
- 1. Through static Fields
- 2. Unclosed Resources
- 3. Improper equals() and hashCode() Implementations
- How to detect a memory leak in Java
- 1. Using Memory Profilers
- 2. Verbose Garbage Collection
- 3. Using Heap Dumps
- Track, Analyze and Manage Java Errors With Rollbar
Диагностика утечек памяти в Java
В данной заметке я хочу показать каким образом можно определять и устранять утечки памяти в Java на примере из моей повседневной работы. Мы не будем здесь рассматривать возможные причины появления утечек, об этом будет отдельная статья, так как тема достаточно обширная. Стоит заметить, что речь пойдет о диагностике именно Heap Memory, об утечках в других областях памяти будет отдельная статья.
Инструменты
Для успешной диагностики нам понадобятся два инструмента: Java Mission Control (jmc) и Eclipse Memory Analyzer. Вобщем-то можно обойтись только Memory Analyzer, но с JMC картина будет более полной.
Анализ использования памяти
Прежде всего, нужно запустить приложение со следующими флагами JVM:
-XX:+UnlockCommercialFeatures
-XX:+FlightRecorder
Не используйте эти опции на production системе без приобретения специальной лицензии Oracle!
Эти опции позволят запустить Flight Recorder – утилита, которая поможет собрать информацию об использовании памяти (и много другой важной информации) во время выполнения программы. Я не буду описывать здесь как запустить Flight Recorder, эта информация легко гуглится. В моем случае было достаточно запустить FR на 10-11 минут.
Рассмотрим следующий рисунок, на котором показана классическая «пила» памяти, а так же важный сигнал, что что-то не так с использованием памяти:
Можно увидеть, что после каждого цикла очистки памяти, heap все больше заполняется, я выделил это желтым треугольником. «Пила» все время как бы ползет вверх. Это значит, что какие-то объекты не достижимы для очистки и накапливаются в old space, что со временем приведет к переполнению этой области памяти.
Выявление утечки
Следующим шагом нужно выявить, что именно не доступно для очистки и в этом нам поможет Memory Analyzer. Прежде всего, нужно загрузить в программу heap dump работающего приложения с предполагаемой утечкой памяти. Это можно сделать с помощью «File → Acquire Heap Dump». После загрузки в диалоге «Getting Started Wizard» выбрать «Leak Suspects Report» после этого откроется краткий обзор возможных утечек памяти:
Если вернуться на вкладку «Overview» и выбрать «Dominator Tree», то можно увидеть более подробную картину:
Дерево показывает структуру «тяжелого» объекта, а так же размер его полей (по типу). Можно видеть, что одно из полей объекта MasterTenant занимает более 45% памяти.
Устранение утечки
Имея результат анализа из предыдущего пункта, следующим шагом идет устранение накапливания объектом памяти. Тут все сильно зависит от конкретного кода. Общая рекоменация – нужно найти и проанализировать все места, где происходит инициализация или изменение соответствующего поля или полей, чтобы понять механизм накапливания памяти. В моем случае в коллекцию постоянно добавлялись записи из множества (около 150) потоков при определенных условиях.
После находжения и устранения утечки, не лишним будет пройти все шаги снова, проанализировать память и отчет Memory Analyzer, чтобы убедиться что фикс помог.
How to Detect Memory Leaks in Java: Causes, Types, & Tools
A memory leak is a situation where unused objects occupy unnecessary space in memory. Unused objects are typically removed by the Java Garbage Collector (GC) but in cases where objects are still being referenced, they are not eligible to be removed. As a result, these unused objects are unnecessarily maintained in memory.
Memory leaks block access to resources and cause an application to consume more memory over time, leading to degrading system performance. If memory leaks are not handled, an application can eventually run out of memory and terminate with an ‘OutOfMemoryError’, which is irrecoverable.
What Causes Memory Leaks in Java
In general, a Java memory leak happens when an application unintentionally (due to logical errors in code) holds on to object references that are no longer required. These unintentional object references prevent the built-in Java garbage collection mechanism from freeing up the memory consumed by these objects.
Common causes for these memory leaks are:
- Excessive session objects
- Insertion without deletion into Collection objects
- Unbounded caches
- Excessive operating system page swapping
- Un-invoked listener methods
- Poorly written custom data structures
Types of Memory Leaks in Java
Memory leaks can be of various types, depending on how they happen. The most common types are detailed below:
1. Through static Fields
Excessive usage of static fields can potentially lead to a memory leak. In Java, static fields usually remain in memory as long as the application is running. Here’s an example:
public class StaticFieldsMemoryLeakExample < private static Listintegers = new ArrayList(); public void insertIntegers() < for (int i = 0; i < 100000000; i++) < integers.add(i); >> public static void main(String[] args) < new StaticFieldsMemoryLeakExample().insertIntegers(); >>
In the above example, the addIntegers() method populates a static List object, which remains in memory throughout the program. The memory usage can be seen in the Java VisualVM monitor below:
As expected, the memory consumed by the List object was not garbage collected and remains in memory.
To prevent these types of memory leaks, the usage of static fields should be minimized, especially when using large objects such as collections. Also, when using singletons, the lazy loading design pattern can be used so that resources are only initialized when they are needed.
2. Unclosed Resources
Resources such as connections and streams utilize memory. If they are not closed, memory allocated to these resources is blocked and the GC is unable to free up this space. Here’s an example:
public void readFromURL() < try < URL url = new URL("http://example.com"); URLConnection urlConnection = url.openConnection(); InputStream is = urlConnection.getInputStream(); byte[] bytes = is.readAllBytes(); >catch (IOException ioe) < ioe.printStackTrace(); >>
In the above example, the readFromURL() method opens a URL connection object but does not close it. Since the object is referenced even after it is no longer used, it continues to block memory and is not eligible for garbage collection. This can be seen in the VisualVM monitor below:
To prevent these types of memory leaks, the finally block should be used to close resources when they are no longer needed. From Java 8 onwards, the try-with-resources block can also be used to automatically close resources. The code to close resources should not itself throw any exceptions.
3. Improper equals() and hashCode() Implementations
Not writing proper overriding implementations for equals() and hashcode() methods when creating a new class can potentially lead to memory leaks in Java. Particularly the HashSet and HashMap classes use these methods for many operations, and it is best to carefully write correct implementations of these methods. Here’s an example of a new class without equals() and hashcode() implementations:
If the above class is used as a key for a HashMap or HashSet, duplicate entries can potentially be added since there is no way to determine how two objects should be considered equal:
public void populateMap() < Mapmap = new HashMap(); for(int i = 0; i < 10000; i++) < map.put(new Foo(1), 1); >>
In the above method, the map object will contain 10000 entries for the same Foo key object, which should only have been inserted once, since Map does not allow duplicate keys. These duplicate objects add up, block memory and are ineligible for garbage collection. This can be seen in the VisualVM monitor below:
As a prevention, when writing new classes, equals() and hashCode() methods should be overridden. Optimally implementing these methods will help with proper utilization of resources and memory.
How to detect a memory leak in Java
Detecting memory leaks requires using a combination of various tools and techniques. Some of the most common and effective ways are:
1. Using Memory Profilers
Memory profilers are tools that can monitor memory usage and help detect memory leaks in an application. Profilers can also help with analyzing how resources are allocated within an application, for example how much memory and CPU time is being used by each method. This can help identify and narrow down any issues.
There are several tools that can be used for memory profiling in Java. Some of the most commonly used ones are — Java VisualVM, JProfiler and YourKit. Java VisualVM was used in this article to help display memory usage in the memory leak examples and illustrate how memory can be monitored using a profiler.
2. Verbose Garbage Collection
To obtain a detailed trace of the Java GC, verbose garbage collection can be enabled. The following parameter can be added to the JVM configuration to enable verbose garbage collection:
The default error output shows the summary, which can help understand how memory is managed and identify any memory leaks.
3. Using Heap Dumps
Heap dumps provide a snapshot of heap memory of a Java application at a particular time. They provide information on how many object instances are open and how much memory they consume. Heap dumps can help with analyzing how many objects are created in an application and if any of them are potentially causing any memory leaks.
Track, Analyze and Manage Java Errors With Rollbar
Managing errors and exceptions in your code is challenging. It can make deploying production code an unnerving experience. Being able to track, analyze, and manage Java errors in real-time can help you to proceed with more confidence. Rollbar automates error monitoring and triaging, making fixing errors easier than ever. Try it today!
«Rollbar allows us to go from alerting to impact analysis and resolution in a matter of minutes. Without it we would be flying blind.»