There are mainly 3 concepts which can break singleton property of a singleton class in java. In this post, we will discuss how it can break and how to prevent those.
Here is sample Singleton class and SingletonTest class.
Singleton.Java
package demo1; public final class Singleton { private static volatile Singleton instance = null; private Singleton() { } public static Singleton getInstance() { if (instance == null) { synchronized (Singleton.class) { if (instance == null) { instance = new Singleton(); } } } return instance; } }
SingletonTest.java
package demo1; public class SingletonTest { public static void main(String[] args) { Singleton object1 = Singleton.getInstance(); Singleton object2 = Singleton.getInstance(); System.out.println("Hashcode of Object 1 - " + object1.hashCode()); System.out.println("Hashcode of Object 2 - " + object2.hashCode()); } }
Here is output, you can see it the same hashcode for objectOne and objectTwo
Hashcode of Object 1 - 1836019240 Hashcode of Object 2 - 1836019240
Now we will break this pattern. First, we will use java reflection.Reflection
ReflectionSingleton.java
Example to show how reflection can break the singleton pattern with Java reflect. You will get two hash code as below. It has a break on the singleton pattern.package demo1; import java.lang.reflect.Constructor; public class ReflectionSingleton { public static void main(String[] args) { Singleton objOne = Singleton.getInstance(); Singleton objTwo = null; try { Constructor constructor = Singleton.class.getDeclaredConstructor(); constructor.setAccessible(true); objTwo = (Singleton) constructor.newInstance(); } catch (Exception ex) { System.out.println(ex); } System.out.println("Hashcode of Object 1 - "+objOne.hashCode()); System.out.println("Hashcode of Object 2 - "+objTwo.hashCode()); } }
Hashcode of Object 1 - 1836019240 Hashcode of Object 2 - 325040804
Prevent Singleton pattern from ReflectionThere are many ways to prevent Singleton pattern from Reflection API, but one of the best solutions is to throw run time exception in the constructor if the instance already exists. In this, we can not able to create a second instance.
private Singleton() { if( instance != null ) { throw new InstantiationError( "Creating of this object is not allowed." ); } }
Deserialization
In serialization, we can save the object of a byte stream into a file or send over a network. Suppose if you serialize the Singleton class and then again de-serialize that object will create a new instance, hence deserialization will break the Singleton pattern.Below code is to illustrate how the Singleton pattern breaks with deserialization.
Implements Serializable interface for Singleton Class.
DeserializationSingleton.Java
package demo1; import java.io.*; public class DeserializationSingleton { public static void main(String[] args) throws Exception { Singleton instanceOne = Singleton.getInstance(); ObjectOutput out = new ObjectOutputStream(new FileOutputStream("file.text")); out.writeObject(instanceOne); out.close(); ObjectInput in = new ObjectInputStream(new FileInputStream("file.text")); Singleton instanceTwo = (Singleton) in.readObject(); in.close(); System.out.println("hashCode of instance 1 is - " + instanceOne.hashCode()); System.out.println("hashCode of instance 2 is - " + instanceTwo.hashCode()); } }
hashCode of instance 1 is - 2125039532 hashCode of instance 2 is - 381259350
Prevent Singleton Pattern from Deserialization
To overcome this issue, we need to override readResolve() method in Singleton class and return same Singleton instance. Update Singleton.java, with below method.
protected Object readResolve() { return instance; }
Now run above DeserializationDemo class and see the output.
hashCode of instance 1 is - 2125039532 hashCode of instance 2 is - 2125039532
Cloning
Using the "clone" method we can create a copy of original object, samething if we applied clone in singleton pattern, it will create two instances one original and another one cloned object. In this case will break Singleton principle as shown in below code.Implement the "Cloneable" interface and override the clone method in the above Singleton class.
Singleton.java
@Override protected Object clone() throws CloneNotSupportedException { return super.clone(); }
Then Test with cloning for breaking the singleton
CloningSingleton.java
public class CloningSingleton { public static void main(String[] args) throws CloneNotSupportedException, Exception { Singleton instanceOne = Singleton.getInstance(); Singleton instanceTwo = (Singleton) instanceOne.clone(); System.out.println("hashCode of instance 1 - " + instanceOne.hashCode()); System.out.println("hashCode of instance 2 - " + instanceTwo.hashCode()); } }
Here is the output
hashCode of instance 1 - 1836019240 hashCode of instance 2 - 325040804
If we see the above output, two instances have different hashcodes means these instances are not the same.
Prevent Singleton Pattern from Cloning
In the above code, breaks the Singleton principle i. e created two instances. To overcome the above issue we need to implement/override clone() method and throw an exception CloneNotSupportedException from clone method. If anyone try to create clone object of Singleton, it will throw an exception as see below code.
@Override protected Object clone() throws CloneNotSupportedException { throw new CloneNotSupportedException(); }
Now we can run the CloningSingleton class, it will throw CloneNotSupportedException while creating a clone object of Singleton object.
Well explained . Great article on singleton pattern . There is also good singleton pattern example visit Singleton class example
ReplyDeleteIt is amazing to visit your site. Thanks for sharing this information, this is useful to me...
ReplyDeleteMicroservices Online Training
Microservices Training in Hyderabad
Are you looking for where to buy hemp oil in Australia? Visit Ricks Hemp Oil store to choose from a premium organic range of hemp seed oil products today!
ReplyDeletecbd oil for sale australia
what is hemp oil
medicinal hemp oil
This comment has been removed by the author.
ReplyDeleteYes, Nowadays mobile application is an Important one, This blog is more informative. Thank you For sharing your knowledge About react Native App Development.
ReplyDeleteNice information. It is very useful for me to learn and understand easily. Thanks for sharing
ReplyDeletereact js online training
Thanks for providing this useful knowledge! In the field of mobile development, Slack clone React Native has a significant presence. And it gets better and better with each new release in terms of development speed and performance. Building a business application used to take a long time, but now that react-native is available, it's possible to construct a messaging platform app in minutes.
ReplyDeletemicroservices
ReplyDeleteTransform your legacy application with a microservices architecture that breaks your application into small components independent from each other.
to get more - https://www.nitorinfotech.com/services/microservices/
Great blog, keep sharing such good work.
ReplyDeleteHire Backend Developer | Hire Developer | Hire Node JS Developer
A very informative blog! \Hire React Developers to create interactive user interfaces using JavaScript library.
ReplyDeleteReally awesome article
ReplyDeleteReact vs React Native
Thanks for sharing a amazing post.
ReplyDeleteHire Dedicated WordPress Developer
Great post on ReactJS and the Virtual DOM! The explanation of how these technologies work together is very insightful. On a related note, if anyone is interested in Full-stack web development USA, I found a fantastic resource that offers comprehensive web development services. You can learn more about it here. Thanks for sharing this informative content!
ReplyDeleteVisit https://l4rgdigitalplus.com/web-development-services-in-usa