Prototype¶
What ?¶
The Prototype Pattern is a creational design pattern used when the cost of creating a new object is expensive or complicated. Instead of creating new instances from scratch, this pattern suggests cloning existing objects to produce new ones.
The pattern allows cloning or copying existing instances to create new ones, ensuring that new objects are created without going through the expensive or complex instantiation process repeatedly.
Key Characteristics
- Objects are created by copying or cloning an existing prototype object.
- Each prototype maintains its state, and the cloned objects are independent copies.
- Useful when the creation of an object is resource-intensive (e.g., large data loads, configurations, etc.).
- Shallow Copy: Only the references of the fields are copied, not the objects themselves.
- Deep Copy: A full independent copy of all nested objects is created.
When to Use ?¶
- When creating an object is expensive e.g., reading a database, complex object initialization, or network-intensive setup.
- When you want to avoid creating new instances and prefer copying existing objects.
- When your code requires many objects of a similar type, where only slight modifications are made to the original object.
- When the class structure or object creation is complex and involves multiple configurations (e.g., with builders, factories, etc.).
- When you need stateless beans that are short-lived and used temporarily.
- When objects need to be created dynamically (e.g., user session-based objects).
Where Not to Use ?¶
- When creating a new instance is inexpensive and straightforward.
- When object state changes frequently, as the cloned object may quickly become stale or out-of-sync with the original.
- If immutable objects are required, as the whole point of this pattern is to create modifiable, independent copies.
- For singleton beans or objects that must be shared across the application.
- When object creation is cheap, and cloning provides no significant benefit.
How to Implement ?¶
Simple Example
In Java, the Prototype Pattern is implemented by making the class implement the Cloneable
interface and overriding the clone()
method from the Object
class.
class Address {
String street;
String city;
public Address(String street, String city) {
this.street = street;
this.city = city;
}
// Deep Copy
public Address(Address address) {
this.street = address.street;
this.city = address.city;
}
@Override
public String toString() {
return street + ", " + city;
}
}
class Employee implements Cloneable {
String name;
Address address;
public Employee(String name, Address address) {
this.name = name;
this.address = address;
}
// Shallow Copy
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
// Deep Copy
public Employee deepClone() {
return new Employee(this.name, new Address(this.address));
}
@Override
public String toString() {
return "Employee: " + name + ", Address: " + address;
}
}
public class PrototypeExample {
public static void main(String[] args) throws CloneNotSupportedException {
Employee emp1 = new Employee("Alice", new Address("123 Street", "New York"));
// Shallow Clone
Employee emp2 = (Employee) emp1.clone();
// Deep Clone
Employee emp3 = emp1.deepClone();
System.out.println("Original: " + emp1);
System.out.println("Shallow Copy: " + emp2);
System.out.println("Deep Copy: " + emp3);
// Modify the original object to see the effect on shallow vs deep copy
emp1.address.street = "456 Avenue";
System.out.println("After modifying the original object:");
System.out.println("Original: " + emp1);
System.out.println("Shallow Copy: " + emp2);
System.out.println("Deep Copy: " + emp3);
}
}
Output
Original: Employee: Alice, Address: 123 Street, New York
Shallow Copy: Employee: Alice, Address: 123 Street, New York
Deep Copy: Employee: Alice, Address: 123 Street, New York
After modifying the original object:
Original: Employee: Alice, Address: 456 Avenue, New York
Shallow Copy: Employee: Alice, Address: 456 Avenue, New York
Deep Copy: Employee: Alice, Address: 123 Street, New York
Explanation
- Shallow Copy (
clone()
): Modifying the original affects the copy (since both reference the sameAddress
object). - Deep Copy (
deepClone()
): The copy remains unaffected because it contains a completely newAddress
object.
Spring Boot Example
Spring Framework allows defining prototype-scoped beans. Each time you request a bean with the prototype scope, Spring returns a new instance, effectively following the Prototype Pattern.
How to Use Prototype Scope in Spring Boot¶
- Add
@Scope
annotation to the bean definition. - Use
prototype
scope to ensure each request gets a new object.
Simple Prototype Scope Example in Spring Boot¶
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;
@Component
@Scope("prototype")
public class Employee {
public Employee() {
System.out.println("New Employee instance created.");
}
}
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class EmployeeController {
@Autowired
private Employee employee1;
@Autowired
private Employee employee2;
@GetMapping("/employees")
public String getEmployees() {
return "Employee 1: " + employee1 + " | Employee 2: " + employee2;
}
}
Summary¶
The Prototype Pattern offers an elegant way to clone existing objects, saving the overhead of complex object creation. It fits well when objects are expensive to create or share the same initial configuration. With Java's cloning mechanisms and Spring Boot's prototype scope, it is easy to implement. However, care must be taken when handling deep versus shallow copies, and the pattern should be avoided when objects are inexpensive to create.