The Decorator allow us to attach additional responsibilities to an object. The concept behind this pattern is not difficult to understand, as long as we have already understood other fundamental concepts as inheritance and composition.
Let’s start simple
A decorator is like a shell. In objectland this translates to an object that holds another object inside.
When we want to “talk” with the inner object, first we “talk” with the outer object, which acts as an intermediary. The outer object will then pass the message to the inner object.
By using this “middleman” technique, we can intercept the messages that are sent to, or sent by, the inner object. If we can intercept them, it becomes obvious that we can manipulate them according to our needs.
Can a Decorator decorate a Decorator ?
It surely can, and that’s when things become interesting.
An image will help to illustrate,
Initial situation :
Assembled situation :
In a situation as the one in the image, when we want to reach object X, we first need to communicate with object C (decorator C), which will pass the message to object B (decorator B), which will pass the message to object A (decorator A), and finally it will arrive to object X. Any reply from the object X, will follow the inverse path, and along the way we can manipulate the “conversation”
How do we build this structure ?
To achieve this object structure, we will have to make sure that the object and the decorators are all of the same type. An Interface will help us on that.
As all the objects are of the same type, we can “communicate” with the outer object as if we where dealing with the most inner object.
Our example
We’ve just started our internet text messaging company. We provide our users with 3 service plans. A free one, that allows them to send plain text messages, a 1 dollar plan that allows them to send encrypted messages and a 2 dollar plan that will add an extra encryption layer to the messages.
In our example, the simple encryption is represented by reversing the message, and the extra security layer is represented by codding the reversed message in Base64. This “highly sophisticated encryption” systems :) will be implemented in a class named Enigma.
This example can illustrate the way we can play with decorators, combining them to achieve different results.
The UML for our example,
We will implement a Message interface, which will allow us to group our objects under the same type. We will only need one method for this interface that will be sendMessage()
1 | package DecoratorPattern.ExampleTWO; |
Now we will implement the MessageDecorator, as an abstract class. All other decorators will derive from this class by extending it, and the behaviour will be defined in each decorator subclass.
1 | package DecoratorPattern.MessageExample; |
Next, we will implement our classes. One for each type of message.
The plain text message StandardMessage Class, will act as the base object ( the one that will be decorated according to our needs). This base class will implement the Message Interface directly.
1 | package DecoratorPattern.MessageExample; |
And now, we implement the decorators which will be the ReversedMessage and EncriptedMessage classes. Both will extend MessageDecorator and by being subclasses, they will have access to parent’s methods.
1 | package DecoratorPattern.MessageExample; |
1 | package DecoratorPattern.MessageExample; |
Finally we implement the Enigma class, which is a helper class that takes care of the our “super secure” encryption system. :)
1 | package DecoratorPattern.MessageExample; |
Let’s play
With our classes in place, we can now start to play with them and by combining decorators we can achieve interesting results.
Sending a plain text message is straightforward. All we need to do is to instantiate a new StandardMessage and call the sendMessage method.
1 | String messageToSend = "This is an important message about the decorator pattern"; |
When we want to decorate a message, is just a question of instantiating the kind of message we want to send, passing as argument either the base object or another decorator.
So, if we want to send a Base64 encrypted message ,
1 | String messageToSend = "This is an important message about the decorator pattern"; |
and for a reversed message,
1 | String messageToSend = "This is an important message about the decorator pattern"; |
In this case we can play even further. If we want to send an Base64 encrypted reversed message, we can instantiate an EncriptedMessage passing as argument a ReversedMessage with a StandardMessage as argument.
1 | String messageToSend = "This is an important message about the decorator pattern"; |
We can do another combination, reversing the message first and Base64 encoding it next, and that would become,
1 | String messageToSend = "This is an important message about the decorator pattern"; |
We can even be crazy enough to double reverse a message, although that is useless, as we will get the initial message, but in an experiment there is nothing like being crazy,
1 | String messageToSend = "This is an important message about the decorator pattern"; |
And this are the basics of the Decorator pattern. A very interesting alternative to sub-classing, that allow us to combine things in very creative ways.
An example for the code in this article can be found in git-hub
Happy coding.