Компоновщик (Composite)

Материал из Вики ИТ мехмата ЮФУ
Версия от 15:24, 3 августа 2014; Admin (обсуждение | вклад) (Описание)

Перейти к: навигация, поиск

К основной странице курса

Назначение

Объединяет объекты в древовидные структуры. Единообразно трактует простые и составные объекты.

Описание

В ряде приложений требуется обрабатывать простые компоненты и составные, состоящие из простых. Например, в векторном графическом редакторе простыми компонентами выступают графические примитивы, а составными - сгруппированный набор графических примитивов.

Необходимость при обработке различать простые и составные компоненты усложняет приложение.

Паттерн Компоновщик рассматривает простые и составные компоненты наследниками некоторого абстрактного класса, тем самым делая единообразной их обработку. Это позволяет в частности включать в составной объект другой составной, формируя дерево объектов.

Реализация

Диаграмма классов

CompositeCommon.png

Участники

Пример

class MainApp
{
    static void Main()
    {
        Composite root = new Composite("root");
        root.Add(new Leaf("Leaf A"));
        root.Add(new Leaf("Leaf B"));

        Composite comp = new Composite("Composite X");                                  
        comp.Add(new Leaf("Leaf XA"));
        comp.Add(new Leaf("Leaf XB"));

        root.Add(comp);
        root.Add(new Leaf("Leaf C"));

        root.Display(1);
    }
}

abstract class Component
{
    protected string name;

    public Component(string name)
    {
        this.name = name;
    }

    public abstract void Add(Component c);
    public abstract void Remove(Component c);
    public abstract void Display(int depth);
}

class Composite : Component
{
    List<Component> children = new List<Component>();

    // Constructor
    public Composite(string name): base(name)
    {  }

    public override void Add(Component component)
    {
        children.Add(component);
    }

    public override void Remove(Component component)
    {
        children.Remove(component);
    }

    public override void Display(int depth)
    {
        Console.WriteLine(new String('-', depth) + name);
        foreach (Component component in children)
            component.Display(depth + 2);
    }
}

class Leaf : Component
{
    public Leaf(string name): base(name)
    {  }

    public override void Add(Component c)
    {
        throw new Exception("Cannot add to a leaf");
    }

    public override void Remove(Component c)
    {
        throw new Exception("Cannot remove from a leaf");
    }

    public override void Display(int depth)
    {
        Console.WriteLine(new String('-', depth) + name);
    }
}

Достоинства и недостатки

Варианты