Composite Design Pattern
Pattern
Composite is a structural design pattern that lets you compose objects into tree structures to represent whole-part hierarchies.
Suppose we have created a simple shape drawing application that can draw basic geometric shapes. Suppose, further, the following is a partial class diagram for this system.
Problem: We want our drawing editor to support grouping and ungrouping operations so that several shapes can be collected together and treated as a single entity.
Solution 1: We could add a group member field into AbstractShape
to indicate which group each shape belongs to (using the number $-1$ to indicate that the object is not in any group).
Pros/Cons
- Pros: simple
- Cons: cannot support nested groups
Solution 2: Introduce a new class ShapeGroup
to manage a group of shapes.
Pros/Cons
- Pros: Supports nested groups
- Cons: You would have to treat primitives (
shapes
) and container classes (shapeGroups
) differently, making the application more complex than necessary.
Solution 3: Implement the ShapeGroup
class as a subclass of Shape
.
The ShapeGroup
class provides a means by which several shapes can be grouped into a single entity that behaves in the same way as a single shape.
The composite design pattern provides a means of grouping together several objects of type $T$ such that the grouped object is also of type $T$.
In the diagram above
Component
:- Declares the abstraction for objects in the composition.
- Could optionally implement default behavior common to all subclasses.
Composite
:- Represents an aggregate object (collection of components).
- Usually has methods to add/remove components.
- Typically implements operations of
Component
simply by calling the same operation for each of its constituent components.
ConcreteComponent
(a.k.a Leaf)- Defines behavior for primitive objects in the composition.
When to use this pattern?
- Need to manipulate a hierarchical collection of "primitive" and "aggregate" objects.
- Need to process (treat) primitive objects the same way as aggregate objects.
- Examples: file/folder hierarchies, organizational charts, tables of contents, parsing structured documents like XML/HTML, etc.
Advantage
Clients typically only need to interact with the
Component
interface; they would be unaware if they're dealing with a composite or a leaf object. This hides the complexity of the data structure from the outside world.