The Best Way to Create Singleton Design Pattern in JavaScript and TypeScript
Add to your RSS feed22 July 20245 min readTable of Contents
What is The Singleton Design Pattern?
The Singleton Design Pattern ensures that a class has only one instance and provides a global point of access to that instance. This pattern is useful when exactly one object is needed to coordinate actions across a system.
The Singleton Design Pattern is used for several key reasons in software development. Here are some of the main advantages and use cases for employing this pattern:
Advantages of Singleton
1. Controlled Access to Sole Instance
-
Ensures Single Instance: Guarantees that a class has only one instance and provides a global point of access to it.
-
Consistency: Helps maintain a single source of truth, preventing inconsistencies and conflicts that can arise from having multiple instances.
2. Resource Management
-
Resource Intensive: Ideal for managing resources such as database connections, file handles, or network connections, where multiple instances would be wasteful or problematic.
-
Improves Performance: Reduces the overhead associated with creating and destroying instances frequently.
Creating a Singleton pattern in JavaScript and TypeScript can be done in several ways. Here’s a detailed guide on how to implement a Singleton pattern effectively:
3. Global State Management
- Centralized Configuration: Useful for managing global states or configurations that need to be consistent across the application. * State Sharing: Facilitates sharing of data or state across different parts of an application without the need to pass instances around.
4. Simplifies Code Structure
- Encapsulation: Encapsulates the instantiation logic within the class, making the codebase easier to understand and maintain. * Avoids Global Variables: Reduces the need for global variables by providing a controlled access point.
5. Thread Safety
- Concurrency Handling: In multi-threaded environments, the Singleton pattern can be designed to be thread-safe, ensuring that only one instance is created even when accessed concurrently.
Use Cases for Singleton Pattern:
1. Logging Services:
- Ensure that there is a single logger instance that handles all log messages.
2. Configuration Management:
- Manage application-wide configurations and settings from a single point.
3. Database Connections:
- Maintain a single connection to a database, ensuring efficient resource usage.
4. Cache Management:
- Implement a cache that is accessed globally and avoids the overhead of creating multiple cache instances.
5. Thread Pool Management:
- Manage a pool of threads or worker objects, ensuring that the pool is created only once.
JavaScript Singleton
Using a ES6 classes:
1 class Singleton {2 constructor() {3 if (!Singleton._instance) {4 Singleton._instance = this;5 }6 return Singleton._instance;7 this.data = 'Hello from Singleton';8 }910 getData() {11 return this.data;12 }13 }1415 const instance1 = new Singleton();16 const instance2 = new Singleton();1718 console.log(instance1 === instance2); // true
Using a Closure:
1 const Singleton = (function () {2 let _instance;34 function createInstance() {5 const object = new Object('Hello from Singleton');6 return object;7 }89 return {10 getInstance: function () {11 if (!_instance) {12 _instance = createInstance();13 }14 return _instance;15 },16 };17 })();1819 const instance1 = Singleton.getInstance();20 const instance2 = Singleton.getInstance();2122 console.log(instance1 === instance2); // true
TypeScript Singleton
Using a TypeScript Class with a Static Property:
1 class Singleton {2 private static _instance: Singleton;34 private constructor(args: string[]) {5 // Private constructor to prevent instantiation6 }78 public someMethod(): string {9 return 'I am the instance';10 }1112 public static getInstance(args: string[]): Singleton {13 if (this._instance) {14 return this._instance;15 }1617 this._instance = new Singleton(args);18 return this._instance;19 }20 }2122 const instance1 = Singleton.getInstance();23 const instance2 = Singleton.getInstance();2425 console.log(instance1 === instance2); // true
Key Points:
-
Singleton Pattern ensures that a class has only one instance and provides a global point of access to it.
-
In TypeScript, you can use the private constructor and static methods to enforce the Singleton pattern.
-
For JavaScript, closures and ES6 classes provide a straightforward way to implement this pattern.
Singleton in older versions of JavaScript
In older versions of JavaScript, you can create a singleton by using an Immediately Invoked Function Expression (IIFE). This pattern leverages closures to create a single instance of an object while keeping private variables and methods out of the global scope.
Here’s how you can create a singleton in ES5-style JavaScript:
1 var Singleton = (function () {2 // Private variable to hold the instance3 var instance;45 // Private method and properties6 function createInstance() {7 var obj = new Object("I am the singleton instance!");8 return obj;9 }1011 // Public API for Singleton12 return {13 getInstance: function () {14 if (!instance) {15 instance = createInstance(); // Create the instance only once16 }17 return instance;18 }19 };20 })();2122 // Usage:23 var singleton1 = Singleton.getInstance();24 var singleton2 = Singleton.getInstance();2526 console.log(singleton1 === singleton2); // Output: true27 console.log(singleton1); // Output: I am the singleton instance!
Explanation
-IIFE: The (function () { ... })()
syntax allows us to create an immediately invoked function that runs once and returns an object containing the getInstance
method. This method provides controlled access to the singleton instance.
- Private
instance
Variable: Since theinstance
variable is within the closure of the IIFE, it’s private and can only be accessed throughgetInstance()
. - Controlled Instance Creation:
getInstance()
checks ifinstance
isundefined
. If it is, it callscreateInstance()
to create and assign the singleton instance; otherwise, it simply returns the existing instance.
This approach ensures that only one instance of the singleton is created, and any subsequent calls will return the same instance.
Example Use Case: Database Connection Pool
1 class DatabaseConnection {2 private static _instance: DatabaseConnection;34 private constructor(args: string[]) {5 // Initialize connection6 }78 public connect() {9 // Connect to the database10 console.log('Database connected');11 }1213 public static getInstance(args: string[]): DatabaseConnection {14 if (this._instance) {15 return this._instance;16 }1718 this._instance = new DatabaseConnection(args);19 return this._instance;20 }21 }2223 // Usage24 const db1 = DatabaseConnection.getInstance();25 const db2 = DatabaseConnection.getInstance();2627 db1.connect();28 console.log(db1 === db2); // true, same instance
Considerations:
-
Overuse Risk: Overusing Singleton can lead to tightly coupled code and make unit testing difficult.
-
Testing Challenges: Singletons can be hard to test due to their global state and potential side effects.
In summary, the Singleton pattern is powerful for managing global states and resources, ensuring consistency and efficient resource management across an application. However, it should be used judiciously to avoid issues related to maintainability and testability.