DEV Community

Cover image for JavaScript Singleton Pattern: Managing a Single Elevator Instance
Vedant Bhamare
Vedant Bhamare

Posted on

JavaScript Singleton Pattern: Managing a Single Elevator Instance

Introduction

Design patterns are essential tools in software development that provide proven solutions to recurring problems. One such pattern is the Singleton Design Pattern. This pattern is used when a system should have exactly one instance of a class, ensuring that the instance is easily accessible from anywhere in the application. In this blog post, we will explore the need for the Singleton Design Pattern, discuss the problems it can solve, and provide a practical example by simulating an elevator system.

Why Do We Need Singleton Design Pattern?

The Singleton Design Pattern is needed in several scenarios where you want to control access to a shared resource or ensure that a particular class has only one instance. Here are some common reasons for using it:

  1. Global Point of Access: When you want a single point of access to an instance, such as a configuration manager or a logging service, the Singleton pattern provides a convenient way to achieve this.
  2. Resource Management: In situations where you need to manage limited resources, like database connections, network sockets, or hardware devices, the Singleton pattern can help ensure that only one instance manages these resources.
  3. System-Wide Objects: If you want a single instance to coordinate actions across your application, such as a print spooler or a thread pool, the Singleton pattern ensures that only one instance is responsible for these tasks.
  4. Lazy Initialization: Singleton allows you to initialize an object only when it is first required, reducing memory consumption and improving performance in scenarios where the object is rarely used.

Common Problems Solved by Singleton Design Pattern

  1. Multiple Instances: Without the Singleton pattern, multiple instances of a class can be created, leading to inconsistencies, increased memory usage, and inefficiencies.
  2. Global State: When you need to maintain a global state, such as configuration settings or shared caches, managing this state through a Singleton ensures a single point of control and consistency.
  3. Thread Safety: In a multi-threaded environment, without Singleton, multiple threads might create multiple instances, which can lead to concurrency issues. The Singleton pattern ensures that only one instance is shared across threads, maintaining thread safety.
  4. Resource Contention: In scenarios where resource contention is a concern (e.g., a single database connection), the Singleton pattern helps to ensure that access is serialized, preventing resource conflicts.

Solving Elevator System Challenges with Singleton

Now, let's dive into a practical example to illustrate the need for the Singleton Design Pattern and how it can solve a common problem. We'll simulate an elevator system for a building where we need to ensure there is only one elevator available.

Problem Statement:

Imagine a building with multiple floors, and we need to manage a single elevator for the entire building. Without the Singleton pattern, it is possible that multiple elevator instances are created, leading to confusion and inefficiency in elevator management.

Solution:

We can implement the Singleton Design Pattern to ensure there is only one elevator in the building. Here's how we can achieve this in code (in a simplified form):

1. Elevator class

class Elevator {
  constructor() {
    this.currentFloor = 1;
  }

  operateElevator(floor) {
    console.log(`Elevator is moving to floor ${floor}`);
    this.currentFloor = floor;
  }
}

Enter fullscreen mode Exit fullscreen mode

This is the Elevator class. It represents the elevator with a property currentFloor to keep track of its current position. It has a method operateElevator that simulates moving the elevator to a given floor by updating the currentFloor property.

2. ElevatorManager Class:

class ElevatorManager {
  constructor() {
    if (!ElevatorManager.instance) {
      ElevatorManager.instance = new Elevator();
    }
    return ElevatorManager.instance;
  }
}
Enter fullscreen mode Exit fullscreen mode

The ElevatorManager class is used to implement the Singleton pattern. Here's how it works:

  • The constructor checks whether an instance of the Elevator class (representing the elevator itself) has already been created. If it hasn't, it creates a new instance of the Elevator and stores it as ElevatorManager.instance.
  • The crucial part of this Singleton pattern is the constructor's return statement. It returns the existing Elevator instance (if it exists) instead of creating a new one. This ensures that there can only be one instance of the elevator, preventing multiple instances from being created.

3. Usage:

const elevator1 = new ElevatorManager();
const elevator2 = new ElevatorManager();
Enter fullscreen mode Exit fullscreen mode
  1. When you create instances of ElevatorManager, you are, in fact, getting the same instance of the Elevator class due to the Singleton pattern. Both elevator1 and elevator2 reference the same elevator instance.

4. Operating the Elevators:

elevator1.operateElevator(5);
elevator2.operateElevator(8);
Enter fullscreen mode Exit fullscreen mode

You can use the operateElevator method on both elevator1 and elevator2 to simulate the elevators moving to different floors. However, since they are referencing the same elevator instance, the currentFloor property is shared between them. So, when you update the floor for one elevator, it affects both:

  • elevator1.operateElevator(5) moves the elevator to floor 5.
  • elevator2.operateElevator(8) moves the same elevator to floor 8.

5. Console Output:

console.log(elevator1.currentFloor); // Output: 8
console.log(elevator2.currentFloor); // Output: 8
Enter fullscreen mode Exit fullscreen mode

When you check the currentFloor property for both elevator1 and elevator2, you'll find that it's the same value (8), confirming that they are using the same instance of the elevator.

Benefits of Using Singleton in the Elevator System:

  1. Single Elevator: With the Singleton pattern, we guarantee that there is only one elevator instance for the entire building, preventing the creation of multiple, conflicting elevators.
  2. Easy Access: The Singleton pattern provides a convenient way to access the elevator instance from any part of the building without the need for complex coordination.
  3. Resource Efficiency: By having only one elevator instance, we reduce memory consumption and ensure efficient use of resources.

Conclusion

The Singleton Design Pattern is a powerful tool for addressing various challenges in software development, such as ensuring a single instance of a class or managing shared resources. By understanding its use cases and benefits, you can make your code more efficient, maintainable, and less error-prone. The elevator system example demonstrates how Singleton can be applied to solve a real-world problem, ensuring that only one elevator is available in the building, which is essential for efficient and safe operation.

Top comments (0)