Singleton design pattern
Created on: Jan 19, 2025
Singleton design pattern is a creation design pattern where class is restricted to have only one instance in the java virtual machine.
There are two ways we can instantiate Singleton design pattern.
- Early instantiation
- Lazy Instantiation
Early instantiation
class Singleton{ private static volatile Singleton singleton = new Singleton(); private Singleton(){}; public static Singleton getInstance(){ return singleton; } }
There are some Drawback of above implementation.
- Instance is created even when not required.
- It can cause memory leak problem
Lazy Instantiation:
class Singleton { private static Singleton singleton; private Singleton() { } public static Singleton getInstance() { if (singleton == null) { singleton = new Singleton(); } return singleton; } }
Singleton nature in above can be broken down in multiple ways.
- Multi Threading
- Reflection api
- Serialization
- Cloning
Let's check one by one and try to resolve those.
1. Multi Threading
if multiple thread is accessing getInstance at same time, it can create multiple instances. Below is same example
package com.learning.test; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; class Singleton { private static Singleton singleton; private Singleton() { } public static Singleton getInstance() { if (singleton == null) { try{ Thread.sleep(1000); }catch (Exception e){ e.printStackTrace(); } singleton = new Singleton(); } return singleton; } } public class Solution { public static void main(String[] args) { Runnable task = () -> { Singleton singleton = Singleton.getInstance(); System.out.println(singleton.hashCode()); }; ExecutorService executorService = Executors.newFixedThreadPool(5); int iterateCount = 5; while (iterateCount-- > 0) { executorService.submit(task); executorService.submit(task); } executorService.shutdown(); } }
938463199 1909241014 1229394636 1909241014 1909241014 1909241014 1909241014 1909241014 1338996218 1221641224
You can see multiple instances as multiple hashcode
are coming. Let's solve this problem by adding the logic for creating instance in synchrnonised
block. Also we will adding volatile
modifier which will ensure that changes made to the singleton instance by one thread are visible to all thread.
class Singleton { private static volatile Singleton singleton; private Singleton() { } public static Singleton getInstance() { if (singleton == null) { try{ Thread.sleep(1000); }catch (Exception e){ e.printStackTrace(); } synchronized(Singleton.class){ // // making this thread safe if(singleton==null){ // if there is no instance available... create new one singleton = new Singleton(); } } } return singleton; } }
2. Reflection api
Java reflection api can change constructor accessibility and create instance. Let's break singleton nature using reflection api.
import java.lang.reflect.Constructor; class Singleton { private static volatile Singleton singleton; private Singleton() { } public static Singleton getInstance() { if (singleton == null) { synchronized(Singleton.class){ if(singleton==null){ singleton = new Singleton(); } } } return singleton; } } public class Solution { public static void main(String[] args) { Singleton singleton = Singleton.getInstance(); Singleton singleton2 = null; try{ Class<Singleton> singletonClass = Singleton.class; Constructor<Singleton> singletonConstructor = singletonClass.getDeclaredConstructor(); singletonConstructor.setAccessible(true); singleton2 = singletonConstructor.newInstance(); }catch (Exception e){ e.printStackTrace(); } System.out.println("hascode 1 "+ singleton.hashCode()); System.out.println("hascode 2 "+ singleton2.hashCode()); } }
hascode 1 764977973 hascode 2 1915910607
To solve this problem, we can add restriction in contructor
. If a instance is not null
, it will throw exception.
import java.lang.reflect.Constructor; class Singleton { private static volatile Singleton singleton; private Singleton() { if (singleton != null) throw new RuntimeException("Only single instance can be created"); } public static Singleton getInstance() { if (singleton == null) { synchronized(Singleton.class){ if(singleton==null){ singleton = new Singleton(); } } } return singleton; } } public class Solution { public static void main(String[] args) { Singleton singleton = Singleton.getInstance(); Singleton singleton2 = null; try{ Class<Singleton> singletonClass = Singleton.class; Constructor<Singleton> singletonConstructor = singletonClass.getDeclaredConstructor(); singletonConstructor.setAccessible(true); singleton2 = singletonConstructor.newInstance(); }catch (Exception e){ e.printStackTrace(); } System.out.println("hascode 1 "+ singleton.hashCode()); System.out.println("hascode 2 "+ singleton2.hashCode()); } }
ava.lang.reflect.InvocationTargetException at java.base/jdk.internal.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method) at java.base/jdk.internal.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:62) at java.base/jdk.internal.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45) at java.base/java.lang.reflect.Constructor.newInstance(Constructor.java:490) at com.learning.test.Solution.main(Solution.java:38) Caused by: java.lang.RuntimeException: Only single instance can be created at com.learning.test.Singleton.<init>(Solution.java:12) ... 5 more hascode 1 2083562754 Exception in thread "main" java.lang.NullPointerException at com.learning.test.Solution.main(Solution.java:43)
3. Serialization
Still our Singleton
class nature can be brake by exporting object into byte stream and again importing it. Below is the code to illustrate it.
import java.io.*; class Singleton implements Serializable{ private static volatile Singleton singleton; private Singleton() { if (singleton != null) throw new RuntimeException("Only single instance can be created"); } public static Singleton getInstance() { if (singleton == null) { synchronized(Singleton.class){ if(singleton==null){ singleton = new Singleton(); } } } return singleton; } } public class Solution { public static void main(String[] args) { try { Singleton singleton = Singleton.getInstance(); ObjectOutput objectOutput = null; objectOutput = new ObjectOutputStream(new FileOutputStream("out.ser")); objectOutput.writeObject(singleton); objectOutput.close(); ObjectInput in = new ObjectInputStream(new FileInputStream("out.ser")); Singleton singleton2 = (Singleton) in.readObject(); in.close(); System.out.println("instance1 hashCode=" + singleton.hashCode()); System.out.println("instance2 hashCode=" + singleton2.hashCode()); } catch (Exception e) { e.printStackTrace(); } } }
instance1 hashCode=356473385 instance2 hashCode=1364614850
We can solve above problem by adding readResolve
method. It's a special method that the serialization framework looks for at runtime. If it is not present, object is created using the no-argument constructor.
class Singleton implements Serializable{ private static volatile Singleton singleton; private Singleton() { if (singleton != null) throw new RuntimeException("Only single instance can be created"); } public static Singleton getInstance() { if (singleton == null) { synchronized(Singleton.class){ if(singleton==null){ singleton = new Singleton(); } } } return singleton; } protected Object readResolve(){ // prevents serialisation effect return getInstance(); } } public class Solution { public static void main(String[] args) { try { Singleton singleton = Singleton.getInstance(); ObjectOutput objectOutput = null; objectOutput = new ObjectOutputStream(new FileOutputStream("out.ser")); objectOutput.writeObject(singleton); objectOutput.close(); ObjectInput in = new ObjectInputStream(new FileInputStream("out.ser")); Singleton singleton2 = (Singleton) in.readObject(); in.close(); System.out.println("instance1 hashCode=" + singleton.hashCode()); System.out.println("instance2 hashCode=" + singleton2.hashCode()); } catch (Exception e) { e.printStackTrace(); } } }
instance1 hashCode=356473385 instance2 hashCode=356473385
4. Cloning
Cloning can also break singleton nature. Below is code to illustrate above.
import java.io.Serializable; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; class Singleton implements Serializable, Cloneable { private static volatile Singleton singleton; private Singleton() { if (singleton != null) throw new RuntimeException("Only single instance can be created"); } public static Singleton getInstance() { if (singleton == null) { synchronized (Singleton.class) { if (singleton == null) { singleton = new Singleton(); } } } return singleton; } protected Object readResolve() { // prevents serialisation effect return getInstance(); } } public class Solution { public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException { Singleton singleton = Singleton.getInstance(); Method cloneMethod = Object.class.getDeclaredMethod("clone"); cloneMethod.setAccessible(true); Singleton singleton2 = (Singleton) cloneMethod.invoke(singleton); System.out.println("Instance 1 hashcode: " + singleton.hashCode()); System.out.println("Instance 2 hashcode: " + singleton2.hashCode()); } }
Instance 1 hashcode: 1308927845 Instance 2 hashcode: 331844619
Let's fix above program by overriding clone
method.
import java.io.Serializable; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; class Singleton implements Serializable, Cloneable { private static volatile Singleton singleton; private Singleton() { if (singleton != null) throw new RuntimeException("Only single instance can be created"); } public static Singleton getInstance() { if (singleton == null) { synchronized (Singleton.class) { if (singleton == null) { singleton = new Singleton(); } } } return singleton; } protected Object readResolve() { // prevents serialisation effect return getInstance(); } @Override protected Object clone() throws CloneNotSupportedException{ throw new CloneNotSupportedException("Singleton cloning not allowed"); } } public class Solution { public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException { Singleton singleton = Singleton.getInstance(); Method cloneMethod = Object.class.getDeclaredMethod("clone"); cloneMethod.setAccessible(true); Singleton singleton2 = (Singleton) cloneMethod.invoke(singleton); System.out.println("Instance 1 hashcode: " + singleton.hashCode()); System.out.println("Instance 2 hashcode: " + singleton2.hashCode()); } }
Exception in thread "main" java.lang.reflect.InvocationTargetException at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.base/java.lang.reflect.Method.invoke(Method.java:566) at com.learning.test.Solution.main(Solution.java:45) Caused by: java.lang.CloneNotSupportedException: Singleton cloning not allowed at com.learning.test.Singleton.clone(Solution.java:32) ... 5 more
You can see that program is throwing exception when we try to clone and create new instance.
Below are some use case of Singleton design pattern
- Database connections are resource-intensive and should be managed carefully. A Singleton can ensure only one connection pool is created for the entire application.
- Applications often have a global configuration (e.g., API keys, file paths, environment settings) that needs to be shared across the app. Using a Singleton ensures all components access the same configuration settings.
- Cache Management: In large-scale applications, caching data improves performance by avoiding repetitive computations or database queries. A Singleton cache ensures there’s only one cache instance, reducing memory usage and preventing inconsistent data.
- Load Balancer: In distributed systems, a load balancer distributes incoming requests to multiple servers. A Singleton load balancer ensures a consistent strategy and state in balancing loads.