Skip to content

Latest commit

 

History

History
110 lines (90 loc) · 3.39 KB

composite.md

File metadata and controls

110 lines (90 loc) · 3.39 KB

Composite Pattern

Problem

We need to build a hierarchy of tree objects and want to interact with all them the same way, regardless of whether it's a leaf object or not.

Solution

There are three main classes in the Composite pattern: the component, the leaf and the composite classes. The component is the base class and defines a common interface for all the components. The leaf is an indivisible building block of the process. The composite is a higher-level component built from subcomponents, so it fulfills a dual role: is a component and a collection of components. As both composite and leaf classes implement the same interface, the can be used the same way.

Example

We've been asked to build a system that keeps track of the manufacturing of cakes, being a key requirement being able to know how long it takes the task of baking it. Making a cake is a complicated process, as it involves multiple tasks that might be composed of different subtasks. The whole process could be represented in the following tree:

|__ Manufacture Cake
    |__ Make Cake
    |   |__ Make Batter
    |   |   |__ Add Dry Ingredients
    |   |   |__ Add Liquids
    |   |   |__ Mix
    |   |__ Fill Pan
    |   |__ Bake
    |   |__ Frost
    |
    |__ Package Cake
        |__ Box
        |__ Label

In the Composite pattern, we'll model every step in a separate class with a common interface, which will report back how long they take. So we'll define a common base class, Task, which plays the role of component.

class Task
  attr_accessor :name, :parent

  def initialize(name)
    @name = name
    @parent = nil
  end

  def get_time_required
    0.0
  end
end

We can now create the classes in charge of the most basic jobs, this is, leaf classes, like AddDryIngredientsTask:

class AddDryIngredientsTask < Task
  def initialize
    super('Add dry ingredients')
  end

  def get_time_required
    1.0
  end
end

What we need now is a container to deal with complex tasks, which are internally built up of any number of subtasks, but from the outside look like any other Task. We'll create the composite class:

class CompositeTask < Task
  def initialize(name)
    super(name)
    @sub_tasks = []
  end

  def add_sub_task(task)
    @sub_tasks << task
    task.parent = self
  end

  def remove_sub_task(task)
    @sub_tasks.delete(task)
    task.parent = nil
  end

  def get_time_required
    @sub_tasks.inject(0.0) {|time, task| time += task.get_time_required}
  end
end

With this base class we can build complex tasks that behave like a simple one, as it implements the Task interface, and also add subtasks with the method add_sub_task. We'll create the MakeBatterTask

class MakeBatterTask < CompositeTask
  def initialize
    super('Make batter')
    add_sub_task(AddDryIngredientsTask.new)
    add_sub_task(AddLiquidsTask.new)
    add_sub_task(MixTask.new)
  end
end

We must keep in mind that the objects tree may go as deep as we want. MakeBatterTask contains only leaf objects, but we could create a class that contains composite objects and it would behave exactly the same:

class MakeCakeTask < CompositeTask
  def initialize
    super('Make cake')
    add_sub_task(MakeBatterTask.new)
    add_sub_task(FillPanTask.new)
    add_sub_task(BakeTask.new)
    add_sub_task(FrostTask.new)
    add_sub_task(LickSpoonTask.new)
  end
end