Usage examples: The Decorator is pretty standard in Python code, especially in code related to streams.
Identification: Decorator can be recognized by creation methods or constructors that accept objects of the same class or interface as a current class.
Conceptual Example
This example illustrates the structure of the Decorator design pattern. It focuses on answering these questions:
What classes does it consist of?
What roles do these classes play?
In what way the elements of the pattern are related?
main.py: Conceptual example
classComponent():""" The base Component interface defines operations that can be altered by decorators. """defoperation(self) ->str:passclassConcreteComponent(Component):""" Concrete Components provide default implementations of the operations. There might be several variations of these classes. """defoperation(self) ->str:return"ConcreteComponent"classDecorator(Component):""" The base Decorator class follows the same interface as the other components. The primary purpose of this class is to define the wrapping interface for all concrete decorators. The default implementation of the wrapping code might include a field for storing a wrapped component and the means to initialize it. """ _component: Component =Nonedef__init__(self,component: Component) ->None: self._component = component@propertydefcomponent(self) -> Component:""" The Decorator delegates all work to the wrapped component. """return self._componentdefoperation(self) ->str:return self._component.operation()classConcreteDecoratorA(Decorator):""" Concrete Decorators call the wrapped object and alter its result in some way. """defoperation(self) ->str:""" Decorators may call parent implementation of the operation, instead of calling the wrapped object directly. This approach simplifies extension of decorator classes. """returnf"ConcreteDecoratorA({self.component.operation()})"classConcreteDecoratorB(Decorator):""" Decorators can execute their behavior either before or after the call to a wrapped object. """defoperation(self) ->str:returnf"ConcreteDecoratorB({self.component.operation()})"defclient_code(component: Component) ->None:""" The client code works with all objects using the Component interface. This way it can stay independent of the concrete classes of components it works with. """# ...print(f"RESULT: {component.operation()}", end="")# ...if__name__=="__main__":# This way the client code can support both simple components... simple =ConcreteComponent()print("Client: I've got a simple component:")client_code(simple)print("\n")# ...as well as decorated ones.## Note how decorators can wrap not only simple components but the other# decorators as well. decorator1 =ConcreteDecoratorA(simple) decorator2 =ConcreteDecoratorB(decorator1)print("Client: Now I've got a decorated component:")client_code(decorator2)
Output.txt: Execution result
Client: I've got a simple component:
RESULT: ConcreteComponent
Client: Now I've got a decorated component:
RESULT: ConcreteDecoratorB(ConcreteDecoratorA(ConcreteComponent))