composite模式(摘自博客园博客)

一、组合模式简介(Brief Introduction

组合模式,将对象组合成树形结构以表示“部分-整体”的层次结构,组合模式使得用户对单个对象和组合对象的使用具有一致性。

 

二、解决的问题(What To Solve

解决整合与部分可以被一致对待问题。

三、组合模式分析(Analysis

1、组合模式结构

 

Component类:组合中的对象声明接口,在适当情况下,实现所有类共有接口的行为。声明一个接口用于访问和管理Component的子部件

Leaf类:叶节点对象,叶节点没有子节点。由于叶节点不能增加分支和树叶,所以叶节点的Add和Remove没有实际意义。

有叶节点行为,用来存储叶节点集合

Composite类:实现Componet的相关操作,比如Add和Remove操作。

children:用来存储叶节点集合

2、源代码

1、抽象类Component

public 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 Diaplay(int depth);

}

 

2、叶子节点Leaf 继承于Component

public class Leaf:Component

{

 

    public Leaf(string name)

        :base(name)

    {

       

    }

 

    public override void Add(Component c)

    {

        Console.WriteLine("不能向叶子节点添加子节点");

    }

 

    public override void Remove(Component c)

    {

        Console.WriteLine("叶子节点没有子节点");

    }

 

    public override void Diaplay(int depth)

    {

        Console.WriteLine(new string('-',depth)+name);

    }

}

 

3、组合类Composite继承于Component,拥有枝节点行为

public class Composite : Component

{

 

    List<Component> children;

 

    public Composite(string name)

        :base(name)

    {

        if (children == null)

        {

            children = new List<Component>();

        }

    }

 

    public override void Add(Component c)

    {

        this.children.Add(c);

    }

 

    public override void Remove(Component c)

    {

        this.children.Remove(c);

    }

 

    public override void Diaplay(int depth)

    {

        Console.WriteLine(new String('-',depth)+name);

        foreach (Component component in children)

        {

            component.Diaplay(depth + 2);

        }

    }

}

 

 

4、客户端代码

static void Main(string[] args)

{

    Composite root = new Composite("根节点root");

    root.Add(new Leaf("根上生出的叶子A"));

    root.Add(new Leaf("根上生出的叶子B"));

 

    Composite comp = new Composite("根上生出的分支CompositeX");

    comp.Add(new Leaf("分支CompositeX生出的叶子LeafXA"));

    comp.Add(new Leaf("分支CompositeX生出的叶子LeafXB"));

 

    root.Add(comp);

 

    Composite comp2 = new Composite("分支CompositeX生出的分支CompositeXY");

    comp2.Add(new Leaf("分支CompositeXY生出叶子LeafXYA"));

    comp2.Add(new Leaf("分支CompositeXY生出叶子LeafXYB"));

 

    comp.Add(comp2);

 

    root.Add(new Leaf("根节点生成的叶子LeafC"));

    Leaf leafD = new Leaf("leaf D");

    root.Add(leafD);

    root.Remove(leafD);

    root.Diaplay(1);

    Console.Read();

}

 

 

3、程序运行结果

四.案例分析(Example

1、场景

假设公司组织结构为:

--总结理

----技术部门经理

------开发人员A

------开发人员B

----销售部门经理

总经理直接领导技术部经理和销售部经理,技术部经理直接领导开发人员A和开发人员B。销售部经理暂时没有直接下属员工,随着公司规模增大,销售部门会新增销售员工。计算组织结构的总工资状况。

如下图所示

 

IComponent接口:此接口包括了Component和Composite的所有属性,公司每个角色都有职称Title和工资待遇Salary,Add方法把员工加入到组织团队中。

Component叶子节点:叶节点没有子节点,Add方法实现没有任何意义。

Composite组合类:此类有一个员工集合_listEmployees,Add方法向此集合中添加员工信息。

GetCost方法获得组织结构中的工资待遇总和

2、代码

1、接口IComponent

public interface IComponent   
    {   
        string Title { get; set; }   
        decimal Salary { get; set; }   
        void Add(IComponent c);   
        void GetCost(ref decimal salary);   
    }   
8.    

 

2、叶节点Component 

public class Component : IComponent   
    {   
        public string Title { get; set; }   
        public decimal Salary { get; set; }   
  
        public Component(string Title, decimal Salary)   
        {   
            this.Title = Title;   
            this.Salary = Salary;   
        }   
  
        public void Add(IComponent c)   
        {   
            Console.WriteLine("Cannot add to the leaf!");   
        }   
  
        public void GetCost(ref decimal salary)   
        {   
            salary += Salary;   
        }   
    }   
22.    

 

 

3、组合类Composite 

1.   public class Composite : IComponent   

2.       {   

3.           private List<IComponent> _listEmployees;   

4.     

5.           public string Title { get; set; }   

6.           public decimal Salary { get; set; }   

7.     

8.           public Composite(string Title, decimal Salary)   

9.           {   

10.               this.Title = Title;   

11.               this.Salary = Salary;   

12.               _listEmployees = new List<IComponent>();   

13.           }   

14.     

15.           public void Add(IComponent comp)   

16.           {   

17.               _listEmployees.Add(comp);   

18.           }   

19.     

20.           public void GetCost(ref decimal salary)   

21.           {   

22.               salary += this.Salary;   

23.     

24.              foreach (IComponent component in this._listEmployees)   

25.               {   

26.                   component.GetCost(ref salary);   

27.               }   

28.           }   

29.       }   

 

 

4、客户端代码

static void Main(string[] args)   
        {   
            decimal costCEO = 0.0M;   
            decimal costVPD = 0.0M;   
  
            //Create CEO Node   
            IComponent compCEO = new Composite("CEO", 500000);   
  
            //Create VP-Development and Developer nodes   
            IComponent compVPDev = new Composite("VP-Development", 250000);   
  
            IComponent compDev1 = new Component("Developer1", 75000);   
            IComponent compDev2 = new Component("Developer2", 50000);   
  
            compVPDev.Add(compDev1);   
            compVPDev.Add(compDev2);   
  
            //Create VP-Sales node   
            IComponent compVPSales = new Component("VP-Sales", 300000);   
  
            compCEO.Add(compVPDev);   
            compCEO.Add(compVPSales);   
  
            //Get the cost incurred at the CEO level   
            compCEO.GetCost(ref costCEO);   
  
            Console.WriteLine(String.Format("The Cost incurred at the CEO            level is {0:c} ", costCEO));   
  
            //Get the cost incurred at the VP-Development level   
            compVPDev.GetCost(ref costVPD);   
            Console.WriteLine(String.Format("The Cost incurred at the VP-Development level is {0:c} ", costVPD));   
        }   
33.    

 

 

五、总结(Summary

组合模式,将对象组合成树形结构以表示“部分-整体”的层次结构,组合模式使得用户对单个对象和组合对象的使用具有一致性。解决整合与部分可以被一致对待问题。

posted @ 2013-04-12 19:42  liyunyu1  阅读(230)  评论(0编辑  收藏  举报