Reference Types In Depth.¶
Let's deep dive to understand How memory management, object references, and behaviors work in Java in this article, with a focus on String handling and other reference types like arrays, classes, and wrapper objects.
Intro¶
Primitive types store values directly in stack memory Where as Reference types store references (addresses) to objects located in heap memory, When you assign a reference type (e.g., String
or an array), only the reference (address) is copied, not the actual data. This means multiple references can point to the same object.
String¶
The String
class in Java is a special reference type with some unique behaviors. Strings are immutable once a String
object is created, it cannot be changed. Any modification on a String
results in the creation of a new object in memory.
String Pool (Interned Strings)¶
A special memory area inside the heap used to store string literals. If a string literal like "Hello"
is created, Java first checks the string pool to see if it already exists. If it does, it returns the reference from the pool. If not, the string is added to the pool.
Example
Heap Memory¶
When you use the new
keyword, a new String
object is always created in the heap memory. Even if the same string already exists in the string pool, the new
keyword forces the creation of a separate instance in the heap.
Example
String s1 = new String("Hello"); // creates a new object outside the pool in the heap.
String s2 = "Hello"; // Stored in the String Pool
System.out.println(s1 == s2); // false (different references)
new String()
, Java forces the creation of a new object in heap even if the same string exists in the pool.
Arrays¶
Arrays are reference types, meaning the array variable stores a reference to the memory location where the array data is stored.
Example
int[] arr1 = {1, 2, 3};
int[] arr2 = arr1; // arr2 now references the same array as arr1
arr2[0] = 10; // Modifies the original array
System.out.println(arr1[0]); // Output: 10 (both arr1 and arr2 reference the same array)
How Array References Work:
arr1
andarr2
point to the same memory location in the heap.- If you change the array via
arr2
, the change will reflect inarr1
because both refer to the same object.
Classes and Objects¶
When you create an object using new
, the reference variable points to the object in heap memory.
Example
class Person {
String name;
}
Person p1 = new Person();
p1.name = "Alice";
Person p2 = p1; // p2 points to the same object as p1
p2.name = "Bob";
System.out.println(p1.name); // Output: Bob (both references point to the same object)
How References Work with Objects:
p1
andp2
point to the same Person object. Changing the object viap2
affectsp1
.- This is how shallow copies work, only the reference is copied, not the object itself.
Wrapper Classes¶
Wrapper classes (Integer
, Double
, Boolean
, etc.) wrap primitive types into objects. These are reference types, and Java performs autoboxing/unboxing to convert between primitive types and wrapper objects.
Example
Wrapper Caching
- Java caches
Integer
objects in the range -128 to 127 for performance. - Beyond this range, new objects are created.
Reference and Deep Copy¶
Shallow Copy: Copies only the reference, so both variables refer to the same object.
Example
Deep Copy: Creates a new object with the same data.
Example
Null/NullPointerException
¶
When a reference is not initialized, it holds the value null
. Accessing a field or method on a null
reference throws a NullPointerException
.
Garbage Collection¶
Java uses Garbage Collection to manage memory. When no references point to an object, it becomes eligible for garbage collection.
Example
We will learn about garbage collection more in depth in another article.
Summary¶
Strings are immutable and stored in the String Pool when created with literals, while using new String() creates a separate object. Arrays are reference types, allowing multiple variables to point to the same array object. For classes, objects are referenced in memory, meaning multiple references can point to the same object. Wrapper classes utilize caching for certain ranges, such as Integer values between -128 and 127. Objects become eligible for garbage collection when no active references point to them.
String Pool In Depth¶
The String Pool (also called the intern pool) in Java is implemented using a Hash Table-like data structure internally. Let’s explore the design and behavior behind this structure:
Internals¶
-
Hash Table:
The String Pool uses a Hash Map-like structure internally, where the key is the string literal, and the value is a reference to the interned string in the pool. This ensures fast lookups and deduplication of identical strings because hash-based structures allow O(1) average-time complexity for lookup operations. -
Behavior of String Pool: If a new string literal is created, the pool checks if the string already exists (by computing its hash), If the string exists, it returns the existing reference. If not, the string is added to the pool.
-
Rehashing and Thread Safety:
The interned pool is part of the JVM's heap and managed by the Java String class. Since Java 7, the pool is stored in the heap instead of the PermGen space to allow better memory management. The pool structure is thread-safe, meaning multiple threads can safely use the pool.
Simplified conceptual pseudocode Example
class StringPool {
private static Map<String, String> pool = new HashMap<>();
public static String intern(String str) {
if (pool.containsKey(str)) {
return pool.get(str); // Return existing reference
} else {
pool.put(str, str); // Add to the pool
return str;
}
}
}
String.intern()
, Java interns the string, meaning it adds the string to the pool if it's not already present.
String Pool Usage Example
public class Main {
public static void main(String[] args) {
String s1 = new String("Hello");
String s2 = s1.intern(); // Adds "Hello" to the pool, if not already present
String s3 = "Hello"; // Uses the interned string from the pool
System.out.println(s2 == s3); // true (same reference from the pool)
}
}
Why Use Hash Table ?¶
- A hash table allows constant time complexity (O(1)) for faster lookups.
- Strings are immutable, so duplicate strings are avoided, improving memory usage so efficient meemory management.
- The JVM ensures that the pool is safe for concurrent access, so multiple threads can use the intern pool efficiently.
Key Takeaways
- Since Java 7, the string pool has been moved to the heap, so it grows dynamically with the application’s memory needs.
- Before Java 7, the pool was in PermGen space, which had a fixed size and could lead to
OutOfMemoryError
if too many strings were interned. - Interning Costs, Calling
intern()
on every string can increase the overhead slightly (because of the hash lookup). It’s only beneficial for reducing memory usage when you expect many repeated strings.
Summary¶
The String Pool is implemented using a Hash Table-like data structure, This allows for efficient string reuse through fast lookups and ensures no duplicate literals are created. Strings added via literals or intern()
are stored in the pool, with existing references returned on subsequent requests.