Singleton vs Prototype Design Pattern in C++: Clear Differences Explained

Learn the key differences between the Singleton and Prototype design patterns in C++, including their use cases, object creation, and when to apply each. This guide provides clear code examples for better understanding.

This question often arises among programmers: Isn’t the clone() method of Prototype design pattern, which returns a copy of an already created object, similar to the Singleton design pattern which also returns an existing object?

The concept of returning an already created object in the clone() method might seem similar to the Singleton Design Pattern at first glance, but there are important differences between the two.

Here’s the concise difference between singleton and prototype design pattern, and if you want to read in detail, please read further:

Singleton Vs Prototype: While Singleton always gives you the SAME instance of an object, the Prototype pattern allows the creation of MULTIPLE, INDEPENDENT objects (clones) based on an existing prototype. The key difference is that Prototype creates new instances, while Singleton ensures there is only one object in the system.

Singleton Design Pattern:

  • Purpose: Ensures that only one instance of a class exists throughout the application. If an object is requested, the Singleton will return the same instance every time.
  • How It Works: The Singleton pattern maintains a static reference to its unique instance, and every time you call the static method (e.g., getInstance()), it returns the same object.
class Singleton {
public:
    static Singleton* getInstance() {
        if (!instance) {
            instance = new Singleton();
        }
        return instance;
    }

private:
    Singleton() {}  // Private constructor to prevent multiple instances
    static Singleton* instance;
};

// Initialization of the static member
Singleton* Singleton::instance = nullptr;

Prototype Design Pattern:

  • Purpose: Creates new objects by cloning an existing object (prototype). It allows you to create new instances based on an existing one, without needing to repeat the entire object creation process.
  • How It Works: Instead of always creating a new object from scratch, you clone an existing object (the prototype). The clone() method creates a copy of the prototype, and the new object is independent of the original.

Key Differences:

  1. Number of Instances:
    • Singleton: Only one instance of the object exists. Every call to getInstance() returns the same instance.
    • Prototype: Multiple independent instances can be created by cloning the prototype. Each clone is a separate object, even though it’s based on the same prototype.
  2. Object Creation:
    • Singleton: You get the same object every time, ensuring a single point of access.
    • Prototype: You can create multiple objects (each clone is independent), even though they all start with the same properties as the prototype.
  3. Use Cases:
    • Singleton is used when you want to restrict the creation of objects to a single instance (e.g., database connection manager, logging service).
    • Prototype is used when you need to create multiple objects that share common characteristics but may have different states (e.g., creating different configurations of a game object, product customization).

Example to Highlight the Difference:

Singleton Example:

class Singleton {
public:
    static Singleton& getInstance() {
        static Singleton instance;
        return instance;
    }

private:
    Singleton() {}  // Private constructor
    Singleton(const Singleton&) = delete;  // Prevent copy
    Singleton& operator=(const Singleton&) = delete;  // Prevent assignment
};

int main() {
    Singleton& s1 = Singleton::getInstance();
    Singleton& s2 = Singleton::getInstance();
    
    std::cout << (&s1 == &s2) << std::endl;  // Outputs 1, meaning both are the same instance
    return 0;
}

Prototype Example:

#include <iostream>
#include <memory>

class Car {
public:
    Car(std::string model) : model(model) {}

    // Clone method to create a copy of the current object
    std::shared_ptr<Car> clone() {
        return std::make_shared<Car>(*this);  // Creates a new object based on the current one
    }

    void describe() {
        std::cout << "Car model: " << model << std::endl;
    }

private:
    std::string model;
};

int main() {
    // Create the prototype
    Car prototype("Sedan");

    // Clone it
    std::shared_ptr<Car> car1 = prototype.clone();
    std::shared_ptr<Car> car2 = prototype.clone();

    car1->describe();  // Car model: Sedan
    car2->describe();  // Car model: Sedan

    // Both objects are independent, even though they are based on the same prototype
    return 0;
}

Summary of Differences:

  • Singleton always gives you the same instance of the object, whereas Prototype gives you new instances based on an existing prototype.
  • Prototype allows for the creation of multiple, independent objects (clones), while Singleton ensures there is only one object in the system.

So, while both patterns can involve sharing an object (via a single instance or cloning), the key distinction lies in how many instances you work with and whether the cloned objects are independent of each other.