Net设计模式实例之组合模式(Composite Pattern)

简介: 一、组合模式简介(Brief Introduction) 组合模式,将对象组合成树形结构以表示“部分-整体”的层次结构,组合模式使得用户对单个对象和组合对象的使用具有一致性。   二、解决的问题(What To Solve) 解决整合与部分可以被一致对待问题。

一、组合模式简介(Brief Introduction

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

 

二、解决的问题(What To Solve

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

三、组合模式分析(Analysis

1、组合模式结构

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

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

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

Composite类:实现Componet的相关操作,比如AddRemove操作。

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接口:此接口包括了ComponentComposite的所有属性,公司每个角色都有职称Title和工资待遇SalaryAdd方法把员工加入到组织团队中。

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

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

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

2、代码

1、接口IComponent

  1. public interface IComponent   

  2.     {   

  3.         string Title { getset; }   

  4.         decimal Salary { getset; }   

  5.         void Add(IComponent c);   

  6.         void GetCost(ref decimal salary);   

  7.     }   

8.  

 

 

2、叶节点Component 

  1. public class Component : IComponent   

  2.     {   

  3.         public string Title { getset; }   

  4.         public decimal Salary { getset; }   

  5.   

  6.         public Component(string Title, decimal Salary)   

  7.         {   

  8.             this.Title = Title;   

  9.             this.Salary = Salary;   

  10.         }   

  11.   

  12.         public void Add(IComponent c)   

  13.         {   

  14.             Console.WriteLine("Cannot add to the leaf!");   

  15.         }   

  16.   

  17.         public void GetCost(ref decimal salary)   

  18.         {   

  19.             salary += Salary;   

  20.         }   

  21.     }   

22.    

 

 

3、组合类Composite 

1.   public class Composite : IComponent   

2.       {   

3.           private List<IComponent> _listEmployees;   

4.     

5.           public string Title { getset; }   

6.           public decimal Salary { getset; }   

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、客户端代码

  1. static void Main(string[] args)   

  2.         {   

  3.             decimal costCEO = 0.0M;   

  4.             decimal costVPD = 0.0M;   

  5.   

  6.             //Create CEO Node   

  7.             IComponent compCEO = new Composite("CEO", 500000);   

  8.   

  9.             //Create VP-Development and Developer nodes   

  10.             IComponent compVPDev = new Composite("VP-Development", 250000);   

  11.   

  12.             IComponent compDev1 = new Component("Developer1", 75000);   

  13.             IComponent compDev2 = new Component("Developer2", 50000);   

  14.   

  15.             compVPDev.Add(compDev1);   

  16.             compVPDev.Add(compDev2);   

  17.   

  18.             //Create VP-Sales node   

  19.             IComponent compVPSales = new Component("VP-Sales", 300000);   

  20.   

  21.             compCEO.Add(compVPDev);   

  22.             compCEO.Add(compVPSales);   

  23.   

  24.             //Get the cost incurred at the CEO level   

  25.             compCEO.GetCost(ref costCEO);   

  26.   

  27.             Console.WriteLine(String.Format("The Cost incurred at the CEO            level is {0:c} ", costCEO));   

  28.   

  29.             //Get the cost incurred at the VP-Development level   

  30.             compVPDev.GetCost(ref costVPD);   

  31.             Console.WriteLine(String.Format("The Cost incurred at the VP-Development level is {0:c} ", costVPD));   

  32.         }   

33.  

 

 

五、总结(Summary

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

 

版权

作者:灵动生活 郝宪玮

出处:http://www.cnblogs.com/ywqu

如果你认为此文章有用,请点击底端的【推荐】让其他人也了解此文章,

img_2c313bac282354945ea179a807d7e70d.jpg

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

 

相关文章
|
Java C#
使用C# (.NET Core) 实现适配器模式 (Adapter Pattern) 和外观模式 (Facade Pattern)
本文的概念内容来自深入浅出设计模式一书 现实世界中的适配器(模式) 我带着一个国标插头的笔记本电脑, 来到欧洲, 想插入到欧洲标准的墙壁插座里面, 就需要用中间这个电源适配器. 面向对象的适配器 你有个老系统, 现在来了个新供应商的类, 但是它们的接口不同, 如何使用这个新供应商的类呢? 首先, 我们不想修改现有代码, 你也不能修改供应商的代码.
1773 0
|
C#
使用C# (.NET Core) 实现组合设计模式 (Composite Pattern)
本文的概念性内容来自深入浅出设计模式一书. 本文需结合上一篇文章(使用C# (.NET Core) 实现迭代器设计模式)一起看. 上一篇文章我们研究了多个菜单一起使用的问题. 需求变更 就当我们感觉我们的设计已经足够好的时候, 新的需求来了, 我们不仅要支持多种菜单, 还要支持菜单下可以拥有子菜单.
1464 0
|
Java C# 设计模式
使用C# (.NET Core) 实现迭代器设计模式 (Iterator Pattern)
本文的概念来自深入浅出设计模式一书 项目需求 有两个饭店合并了, 它们各自有自己的菜单. 饭店合并之后要保留这两份菜单. 这两个菜单是这样的: 菜单项MenuItem的代码是这样的: 最初我们是这样设计的, 这是第一份菜单: 这是第2份菜单: 同时有两个菜单存在的问题 问题就是多个菜单把事情变复杂了.
1036 0
|
算法 C# Java
使用C# (.NET Core) 实现模板方法模式 (Template Method Pattern)
本文的概念内容来自深入浅出设计模式一书. 项目需求 有一家咖啡店, 供应咖啡和茶, 它们的工序如下: 咖啡: 茶: 可以看到咖啡和茶的制作工序是差不多的, 都是有4步, 其中有两步它们两个是一样的, 另外两步虽然具体内容不一样, 但是都做做的同一类工作.
1382 0
|
C#
使用C# (.NET Core) 实现命令设计模式 (Command Pattern)
本文的概念内容来自深入浅出设计模式一书. 项目需求 有这样一个可编程的新型遥控器, 它有7个可编程插槽, 每个插槽可连接不同的家用电器设备. 每个插槽对应两个按钮: 开, 关(ON, OFF).
860 0
|
Java C#
使用C# (.NET Core) 实现单体设计模式 (Singleton Pattern)
本文的概念内容来自深入浅出设计模式一书 由于我在给公司做内培, 所以最近天天写设计模式的文章.... 单体模式 Singleton 单体模式的目标就是只创建一个实例. 实际中有很多种对象我们可能只需要它们的一个实例, 例如: 线程池,缓存, 弹出的对话框, 用于保存设置的类, 用于logging的类, 硬件设备驱动对象等等.
1228 0
|
C#
使用C# (.NET Core) 实现抽象工厂设计模式 (Abstract Pattern)
本文的概念性内容来自深入浅出设计模式一书. 上一篇文章讲了简单工厂和工厂方法设计模式 http://www.cnblogs.com/cgzl/p/8760250.html, 使用的是披萨店的例子. 文将继续使用这个例子, 这里要用到抽象工厂.
1373 0
|
C# 设计模式 .NET
使用C# (.NET Core) 实现简单工厂(Simple Factory) 和工厂方法设计模式 (Factory Method Pattern)
本文源自深入浅出设计模式. 只不过我是使用C#/.NET Core实现的例子.   前言 当你看见new这个关键字的时候, 就应该想到它是具体的实现. 这就是一个具体的类, 为了更灵活, 我们应该使用的是接口(interface).
1460 0
|
安全 C# 数据安全/隐私保护
使用C# (.NET Core) 实现装饰模式 (Decorator Pattern) 并介绍 .NET/Core的Stream
该文章综合了几本书的内容. 某咖啡店项目的解决方案 某咖啡店供应咖啡, 客户买咖啡的时候可以添加若干调味料, 最后要求算出总价钱. Beverage是所有咖啡饮料的抽象类, 里面的cost方法是抽象的.
1393 0
|
C#
使用C# (.NET Core) 实现观察者模式 (Observer Pattern) 并介绍 delegate 和 event
观察者模式 这里面综合了几本书的资料. 需求 有这么个项目:  需求是这样的: 一个气象站, 有三个传感器(温度, 湿度, 气压), 有一个WeatherData对象, 它能从气象站获得这三个数据.
1971 0