GOF Design Patterns (2) : Singleton Design Pattern
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 | public class ExampleClass { // Note 1 private static ExampleClass instance; private String someAttribute; // Note 2 private ExampleClass(String someAttribute) { this.someAttribute = someAttribute; } // Note 3 public static ExampleClass getInstance(String someAttribute) { // Note 4 if (instance == null) { instance = new ExampleClass(someAttribute); } // Note 5 return instance } } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 | public class ExampleClass { private static ExampleClass instance; private String someAttribute; private ExampleClass(String someAttribute) { this.someAttribute = someAttribute; } public static ExampleClass getInstance(String someAttribute) { // Note 6 synchronized (ExampleClass.class) { if(instance==null) { instance = new ExampleClass(someAttribute); } } return instance; } } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 | public class ExampleClass { private static ExampleClass instance; private String someAttribute; private ExampleClass(String someAttribute) { this.someAttribute = someAttribute; } public static ExampleClass getInstance(String someAttribute) { // Note 7 if (instance == null) { synchronized(ExampleClass.class) { if (instance == null) { instance = new ExampleClass(someAttribute); } } } return instance; } } |
So, everything is fine?
How do we do this?
- Every read and write happens directly in main memory.
- No thread reads a stale cached value.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 | public class ExampleClass { // Note 8 private static volatile ExampleClass instance; private String someAttribute; private ExampleClass(String someAttribute) { this.someAttribute = someAttribute; } public static ExampleClass getInstance(String someAttribute) { if (instance == null) { synchronized(ExampleClass.class) { if (instance == null) { instance = new ExampleClass(someAttribute); } } } return instance; } } |
To solve this, we can ask the thread to use a local variable to refer to the instance variable in main memory. By using this local variable, the thread can cache it, improving performance while maintaining thread safety.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 | public class ExampleClass { private static volatile ExampleClass instance; private String someAttribute; private ExampleClass(String someAttribute) { this.someAttribute = someAttribute; } public static ExampleClass getInstance(String someAttribute) { ExampleClass localInstance = instance; // Note 9 if (localInstance == null) { // Note 9 synchronized(ExampleClass.class) { localInstance = instance; // Note 9 if (localInstance == null) { // Note 9 instance = localInstance = new ExampleClass(someAttribute); // Note 9 } } } return localInstance; // Note 9 } } |
- Note 9 : Now, we have created a localInstance at line 12 to store a reference to our volatile instance field. This allows threads to cache localInstance while maintaining thread safety. At line 14, we check for null using that local reference. If it's null, we proceed with the synchronized block. After blocking other threads from accessing the synchronized block, we fetch the instance again at line 17 and check for null at line 19 to ensure that no other thread has created an instance in the meantime. Then, we update both the local variable and static volatile variable with our new object reference at line 20. Finally, at line 27, we return the instance to the requester.
Now we got a robust solution for the Singleton Design Pattern with both thread safety and optimized performance! 🚀
When to Use Singleton
- Database Connection Manager : Ensures a single instance of the database connection pool is shared across the application to optimize resource usage.
- Logger Utility : Provides a single logging instance to ensure consistent logging behavior across the application.
- Configuration Manager : Centralizes access to application-wide settings or configurations, avoiding redundant reloading of settings.
- Thread Pool Manager : Manages a shared pool of threads to optimize thread creation and reuse, ensuring efficient multithreading.
- Caching System : Maintains a single instance of a cache (e.g., in-memory cache) to store frequently accessed data for improved performance.
Yup. That's it for Now
Happy Coding 🙌