Skip to content
SP StackPractices
beginner

Singleton Pattern

Ensure a class has only one instance and provide global access to it. A creational design pattern for controlled object creation.

Topics: design

Singleton Pattern

Overview

The Singleton Pattern is a creational design pattern that restricts a class to a single instance and provides a global point of access to it. It is useful when exactly one object is needed to coordinate actions across the system.

Common use cases include database connection pools, configuration managers, and logging services.

When to Use

Use the Singleton Pattern when:

  • Exactly one instance of a class must exist in the system
  • A single shared resource needs controlled access (e.g., config, cache, connection pool)
  • You need a global access point without polluting the namespace with global variables
  • Lazy initialization is desired to avoid creating the instance until it is needed

Solution

Python

class Singleton:
    _instance = None

    def __new__(cls):
        if cls._instance is None:
            cls._instance = super().__new__(cls)
        return cls._instance

# Usage
a = Singleton()
b = Singleton()
print(a is b)  # True

JavaScript

class Singleton {
  static #instance = null;

  static getInstance() {
    if (!Singleton.#instance) {
      Singleton.#instance = new Singleton();
    }
    return Singleton.#instance;
  }
}

// Usage
const a = Singleton.getInstance();
const b = Singleton.getInstance();
console.log(a === b); // true

Java

public class Singleton {
    private static Singleton instance;

    private Singleton() {}

    public static synchronized Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

// Usage
Singleton a = Singleton.getInstance();
Singleton b = Singleton.getInstance();
System.out.println(a == b); // true

Explanation

The Singleton Pattern guarantees a single instance through three mechanisms:

  • Private constructor: Prevents direct instantiation from outside the class
  • Static instance field: Holds the single shared instance
  • Global access method: Provides a controlled way to retrieve the instance

In multi-threaded environments (like Java), use synchronized or eager initialization to prevent race conditions during instance creation.

Variants

VariantUse CaseTrade-off
Lazy initializationInstance created on first accessThread-safety concerns
Eager initializationInstance created at class loadNo thread issues, may waste resources
Double-checked lockingHigh-performance lazy initMore complex, error-prone in some languages

Best Practices

  • Make the constructor private to prevent accidental direct instantiation
  • Use lazy initialization only when startup cost matters
  • Consider thread safety in concurrent environments
  • Avoid overuse: Singletons can make unit testing harder due to hidden global state
  • Document the singleton nature so other developers do not try to create multiple instances

Common Mistakes

  • Race conditions: Two threads creating separate instances simultaneously
  • Testing difficulties: Hidden global state makes tests order-dependent
  • Overuse: Turning every shared service into a singleton increases coupling
  • Serialization issues: Deserializing can create duplicate instances unless handled
  • Inheritance misuse: Subclasses can break the single-instance guarantee

Frequently Asked Questions

Q: Is Singleton an anti-pattern? A: Not inherently, but overuse leads to tight coupling and hidden dependencies. Use it sparingly for true single-instance resources.

Q: How do I make a Singleton thread-safe in Python? A: The __new__ approach shown above is thread-safe in CPython due to the GIL. For stricter safety, use a lock or module-level variables.

Q: Can a Singleton have subclasses? A: It is possible but tricky. Each subclass can end up with its own instance, which may or may not be the desired behavior.