- Published on
Decorator Pattern
- Authors
- Name
- Mustafiz Kaifee
- @mustafiz_kaifee
The Decorator Pattern is a structural design pattern that allows behaviour to be added to specific objects, either statically or dynamically, while preserving the behaviour of other objects in the same class. This pattern offers a more flexible alternative to subclassing for adding functionality.
Current Problem
Subclassing is a common technique used in object-oriented design to extend an object's capabilities. However, subclassing creates rigidity because it is static and applies to the entire class rather to individual objects. It can result in a proliferation of subclasses, making a system too complex and difficult to understand, particularly when adding various functionalities.
Solution through Adapter Pattern
The Decorator Pattern addresses these concerns by allowing additional functionality to be added to objects without inheriting from them. It accomplishes this by defining a collection of decorator classes that are used to surround concrete components. This technique allows you to dynamically and transparently add responsibilities to objects without changing their code.
Real-World Applications
- Graphical User Interface (GUI) Toolkits: Add more functionality to visual components such as borders, scrolling, and colour changes without changing the components themselves.
- Data Stream Enhancements: Adding functions like buffering, compression, or encryption to data streams without altering the underlying data source or the stream's essential functionality.
- Middleware Services: Flexible and modular enhancements to service objects in middleware, such as logging, transaction management, or error handling.
Structure
The Decorator Pattern typically involves the following components:
- Component is an interface for objects that can be dynamically assigned responsibilities.
- ConcreteComponent: A concrete implementation of the Component that defines an object to which extra responsibilities can be assigned.
- Decorator: An abstract class that implements the Component interface and stores a reference to a Component instance.
- ConcreteDecorator: A concrete version of the Decorator that adds duties to the component.
Real-World Example with Code Snippet
Imagine a simple coffee shop application where you can add different ingredients to the base coffee:
// Component
interface Coffee {
double cost();
String description();
}
// ConcreteComponent
class BasicCoffee implements Coffee {
public double cost() {
return 1.0;
}
public String description() {
return "Coffee";
}
}
// Decorator
abstract class CoffeeDecorator implements Coffee {
protected Coffee decoratedCoffee;
public CoffeeDecorator(Coffee coffee) {
this.decoratedCoffee = coffee;
}
}
// ConcreteDecorator
class MilkDecorator extends CoffeeDecorator {
public MilkDecorator(Coffee coffee) {
super(coffee);
}
public double cost() {
return decoratedCoffee.cost() + 0.5;
}
public String description() {
return decoratedCoffee.description() + ", Milk";
}
}
// Main class
public class Main {
public static void main(String[] args) {
Coffee myCoffee = new BasicCoffee();
myCoffee = new MilkDecorator(myCoffee);
System.out.println(myCoffee.description() + ": $" + myCoffee.cost());
}
}
This example demonstrates adding functionality (milk) to a basic coffee object dynamically, showcasing how the Decorator Pattern works.
Pros and cons
Pros:
- Offers more flexibility than static inheritance.
- Eliminates the need for an excessive number of subclasses, making code maintenance easier.
- Enables the dynamic and transparent addition of tasks.
Cons:
- Can result in many little classes, making a system more difficult to understand.
- Overuse can produce complex code that is difficult to debug and maintain.
- The initial setup may be more involved than utilising subclasses.
Conclusion
The Decorator Pattern provides a reliable approach for expanding object functionality in a flexible and scalable way. It enables developers to add responsibilities dynamically and helps keep code simple even if needs change or new features are required.