Memory Management


Table of Contents
  1. Quick Revision
  2. What Kinds of Memory Does the JVM Manage?
  3. Different Parts of the Heap in Java
  4. What is PermGen Space?
  5. What is Metaspace?
  6. Difference Between PermGen and Metaspace
  7. What is Java HotSpot?
  8. What is the Java Interpreter?
  9. Working of the JIT Compiler
  10. What is the Code Cache in Java HotSpot?
  11. Reference Types in Java
  12. What is the Use of ReferenceQueue?
  13. What is a Garbage Collector?
  14. How Garbage Collection Process in Java
  15. Can We Force the Garbage Collector to Run?
  16. What is MinorGC and MajorGC?
  17. What is a memory leak?
  18. How to Generate and Analyze Heap Dump in GraalVM
  19. How you handled Memory leak in your project?

Quick Revision

What Kinds of Memory Does the JVM Manage?

The Java Virtual Machine (JVM) manages several types of memory, categorized into distinct areas:


Different Parts of the Heap in Java

The heap in Java is divided into multiple regions to optimize garbage collection and memory management. Here's a detailed explanation of the parts:

Heap Generations Overview:

Heap Size Limits: The theoretical maximum heap limit for a 32-bit JVM is 4GB, but practical limits are lower due to constraints like swap memory, kernel address space, memory fragmentation, and VM overhead.


What is PermGen Space?

PermGen (Permanent Generation) is a memory pool in the HotSpot JVM used to store metadata about the virtual machine itself and user-defined classes. It holds reflective data, such as class and method objects, and is unique to the HotSpot JVM.

Key Characteristics of PermGen: Relation to Heap: Challenges with PermGen: Configuration Options: Transition to Metaspace:

Starting with Java 8, PermGen was replaced by Metaspace, which uses native memory instead of heap memory. This change resolved many of the limitations of PermGen, such as fixed-size constraints.


What is Metaspace?

With the introduction of JDK 8, the concept of PermGen space has been replaced by Metaspace. The metadata information previously stored in PermGen is now relocated to native memory, under the Metaspace. This transition aligns with practices used by other JVM implementations like Oracle JRockit and IBM JVM.

Key Features of Metaspace:

Difference Between PermGen and Metaspace

The primary difference between PermGen and Metaspace lies in their memory management. While PermGen is contiguous within the Java heap, Metaspace is external to the Java heap and resides in native memory. This separation from the heap means that Metaspace's size is not fixed and instead, it dynamically adjusts its allocation based on the available resources of the host operating system.

Key Differences:

What is Java HotSpot?

Java HotSpot is a high-performance implementation of the Java Virtual Machine (JVM) developed by Oracle Corporation. It enables Java applications to run efficiently across various hardware and operating systems.

Key Features of Java HotSpot

What is the Java Interpreter?

The Java Interpreter is a component of the Java runtime environment that directly executes Java bytecode instructions. Bytecode is the intermediate representation of Java programs, generated by the Java compiler, and is platform-independent.

Key Features of the Java Interpreter

Working of the JIT Compiler

A Just-In-Time (JIT) compiler operates after a program has started execution, dynamically converting bytecode or virtual machine instructions into the host CPU's native machine code. This process occurs "just-in-time" to optimize program performance.

Key Characteristics of JIT Compilation Advantages and Trade-offs

The JIT compiler improves Java program performance to approach that of native applications, but its dynamic nature requires a balance between compilation time, memory usage, and runtime performance. Optimization opportunities are greatest during the program's early stages, with diminishing returns as execution stabilizes.


What is the Code Cache in Java HotSpot?

In order to run JVM bytecode on different platforms, it needs to be converted to machine instructions. The JIT compiler is responsible for this compilation as the program is executed. When the JVM compiles bytecode to assembly instructions, it stores those instructions in a special non-heap data area called Code Cache.


Reference Types in Java

In Java, references are used to access objects stored in memory. The Java programming language provides four types of references to manage memory and the lifecycle of objects, especially in relation to garbage collection.

Types of References Summary
Reference Type Garbage Collection Eligibility Use Case
Strong Reference Not collected until explicitly nullified Default reference type
Weak Reference Collected when no strong references exist Caches, maps
Soft Reference Collected when memory is low Memory-sensitive caches
Phantom Reference Collected after finalize action Post-mortem cleanup

What is the Use of ReferenceQueue?

A ReferenceQueue in Java is a utility class used in conjunction with reference types like WeakReference, SoftReference, and PhantomReference. It provides a mechanism to track when a referent object is about to be or has been reclaimed by the garbage collector.

Key Uses of ReferenceQueue How ReferenceQueue Works

When creating a WeakReference, SoftReference, or PhantomReference, you can associate it with a ReferenceQueue. When the garbage collector determines that the referent object is eligible for collection, the reference object is enqueued in the associated ReferenceQueue.

The ReferenceQueue is a powerful tool for managing object lifecycles and implementing custom memory management strategies. It is particularly useful for advanced use cases like caching, resource cleanup, or integrating garbage collection events with application logic.


What is a Garbage Collector?

In Java, the Garbage Collector (GC) is a component of the Java Virtual Machine (JVM) responsible for automatic memory management. Its primary role is to reclaim memory occupied by objects that are no longer reachable or needed by the application, ensuring efficient use of system resources and preventing memory leaks.

Key Features of Garbage Collection How the Garbage Collector Works

The GC employs algorithms to identify and remove unreachable objects:

Types of Garbage Collectors

Java provides several types of garbage collectors, which can be selected using JVM options:

Advantages

How Garbage Collection Process in Java

Garbage collection in Java is a mechanism to automatically manage memory by reclaiming memory occupied by objects that are no longer reachable. The garbage collection process is integral to the JVM's memory management, involving multiple memory areas and distinct phases.

Memory Structure

The JVM divides the heap memory into the following regions:

Garbage Collection in the Young Generation

The young generation consists of the Eden memory and the two survivor spaces. Here's how the garbage collection process works:

Purpose of Two Survivor Spaces

The two survivor spaces are designed to prevent memory fragmentation. After each garbage collection cycle:

Garbage Collection in the Old Generation

Once objects are promoted to the old generation, garbage collection occurs less frequently but involves more intensive processing:

Conclusion

The garbage collection process efficiently manages memory by reclaiming unused objects, promoting long-lived objects, and optimizing memory layout. It ensures the smooth functioning of Java applications while reducing the need for manual memory management.


Can We Force the Garbage Collector to Run?

In Java, it is not possible to force the garbage collector to run. The methods System.gc() and Runtime.getRuntime().gc() can be used to suggest that garbage collection should be performed, but these are merely hints to the JVM. The decision to perform garbage collection is entirely up to the JVM and its internal algorithms.

Key Points Code Example
  public class GarbageCollectorExample {
      public static void main(String[] args) {
          // Suggesting garbage collection
          System.gc();
          Runtime.getRuntime().gc();
          
          // No guarantee that GC will run immediately
          System.out.println("Garbage collection suggested.");
      }
  }
    
Conclusion

While System.gc() and Runtime.getRuntime().gc() can be used to suggest garbage collection, they do not force it. The JVM's garbage collector operates autonomously to optimize application performance and memory management.


What is MinorGC and MajorGC?

In Java, garbage collection is categorized based on the memory region it operates on. These terms, although commonly used, do not have formal definitions in the JVM specification or garbage collection research papers.

MinorGC MajorGC FullGC Relationship Between MinorGC and MajorGC

MajorGCs are often triggered by MinorGCs. For instance, when surviving objects from the young generation fill the old generation during promotion, a MajorGC may be required. This interdependence makes it challenging to separate the two processes entirely.

Conclusion

While MinorGC and MajorGC are widely used terms, they are not formally defined in the JVM specification. They provide a convenient way to describe garbage collection activity in different regions of the heap, helping developers understand and optimize memory management in Java applications.


What is a memory leak?

A memory leak in Java is a situation where some objects are no longer used by an application, but the Garbage Collector fails to recognize them as unused. This leads to the OutOfMemoryError if those unused objects contribute to the heap usage significantly enough that the next memory allocation request by the application cannot be fulfilled.

🔍 How to Avoid Memory Leaks in Java

How to Generate and Analyze Heap Dump in GraalVM
Enable Heap Dump Generation

Run your Java application with the following JVM option:

-XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=heapdump.hprof

This automatically generates a heap dump when an OutOfMemoryError occurs.

Manually Generate Heap Dump

You can manually create a heap dump using one of the following methods:

Using `jcmd` (Recommended in GraalVM)
jcmd  GC.heap_dump heapdump.hprof
Using `jmap` (Alternative)
jmap -dump:live,format=b,file=heapdump.hprof 
Using GraalVM’s Native Image (If running native binary)

If you are using GraalVM’s **native-image**, enable heap dump support using:

-XX:+HeapDumpAfterFullGC
Analyze Heap Dump with VisualVM

GraalVM provides VisualVM as a profiling tool.

Open Heap Dump in VisualVM
visualvm

Steps:

Analyze Heap Dump with Eclipse MAT (Memory Analyzer Tool)

For advanced memory leak detection:

Fix Memory Issues

Based on the heap dump analysis, you can:

Summary

How you handled Memory leak in your project?

In one of our high-traffic microservices, we encountered an OutOfMemoryError (OOM) in production. The service was responsible for handling real-time user sessions, and over time, it started consuming more memory than expected.

Step 1: Identifying the Issue Initial Debugging

Step 2: Debugging Using Heap Dump
jcmd <PID> GC.heap_dump heapdump.hprof
Findings from Heap Dump Analysis

Step 3: Root Cause Analysis

The UserSessionManager class was using a static HashMap to store user sessions:

public class UserSessionManager {
private static final Map<String, UserSession> sessionCache = new HashMap<>();

public static void addSession(String sessionId, UserSession session) {
    sessionCache.put(sessionId, session);
}
}
    

Step 4: Solution & Fix
1. Used WeakHashMap Instead of HashMap
private static final Map<String, UserSession> sessionCache = new WeakHashMap<>();
    
2. Implemented Session Timeout & Cleanup Mechanism
ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);
scheduler.scheduleAtFixedRate(() -> {
sessionCache.entrySet().removeIf(entry -> entry.getValue().isExpired());
}, 10, 10, TimeUnit.MINUTES);
    
3. Optimized JVM Heap Settings
-Xms512m -Xmx2g
-XX:+UseG1GC -XX:InitiatingHeapOccupancyPercent=45

Step 5: Validation & Results