Debugging Memory Leaks in Java: 3 Practical Examples

Learn how to identify and fix memory leaks in Java applications with these practical examples.
By Jamie

Understanding Memory Leaks in Java

Memory leaks occur when an application continues to hold references to objects that are no longer needed, preventing the garbage collector from reclaiming memory. This can lead to increased memory usage and eventual crashes. Identifying and fixing these leaks is crucial for maintaining application performance. Here are three practical examples of debugging memory leaks in a Java application.

Example 1: Unintentional Static References

In this scenario, a static list is holding references to objects that should be eligible for garbage collection, causing a memory leak.

Consider a web application that caches user sessions in a static list. Every time a user logs in, their session is added to this list. If the application fails to remove sessions after they expire, memory consumption increases over time.

import java.util.ArrayList;
import java.util.List;

public class SessionManager {
    private static List<String> activeSessions = new ArrayList<>();

    public static void addSession(String sessionId) {
        activeSessions.add(sessionId);
    }

    public static void removeSession(String sessionId) {
        activeSessions.remove(sessionId);
    }
}

To fix this, ensure that expired sessions are removed from the list:

public static void removeExpiredSessions(List<String> expiredSessions) {
    for (String sessionId : expiredSessions) {
        removeSession(sessionId);
    }
}

Note: Consider using a weak reference or an expiration policy to avoid retaining unnecessary references.

Example 2: Listener Registration Without Deregistration

Memory leaks can also occur when event listeners are registered but not properly deregistered. This is common in GUI applications where components remain in memory because they still hold references to their listeners.

For instance, in a Java Swing application, if you add an ActionListener to a JButton but never remove it, the JButton will prevent the listener from being garbage collected:

import javax.swing.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

public class ButtonExample {
    private JButton button;

    public ButtonExample() {
        button = new JButton("Click Me");
        button.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {
                // Action code
            }
        });
    }
}

To prevent a memory leak, deregister the listener when the button is no longer needed:

public void cleanup() {
    for (ActionListener listener : button.getActionListeners()) {
        button.removeActionListener(listener);
    }
}

Variation: Use weak references for listeners if applicable, allowing them to be garbage collected when no longer needed.

Example 3: Long-lived Threads Holding References

When using threads, especially in server applications, these can hold references to objects that are no longer required, causing memory leaks.

Imagine a background thread that processes data and retains references to input data it has handled:

public class DataProcessor extends Thread {
    private List<Data> dataList;

    public DataProcessor(List<Data> dataList) {
        this.dataList = dataList;
    }

    @Override
    public void run() {
        while (!Thread.currentThread().isInterrupted()) {
            // Process data
        }
    }
}

If the DataProcessor is not properly stopped or if it holds onto references, it can cause a memory leak.

A better approach is to clear the list of data or stop the thread when it’s no longer needed:

public void stopProcessing() {
    this.interrupt();
    dataList.clear();
}

Note: Always ensure threads are managed properly, including stopping them and releasing resources to prevent memory leaks.

These examples illustrate common situations where memory leaks can occur in Java applications and highlight methods for debugging and resolving these issues. By being mindful of object references, deregistering listeners, and managing threads, developers can significantly reduce the risk of memory leaks in their applications.