概念

组合是一种结构型设计模式, 你可以使用它将对象组合成树状结构, 并且能像使用独立对象一样使用它们。

组合模式(Composite Pattern)是将对象组合成树形结构以表示‘部分-整体’的层次结构。组合模式使得用户对单个对象和组合对象的使用具有一致性。

对于绝大多数需要生成树状结构的问题来说, 组合模式都是非常好的一种解决方案。 主要的功能是在整个树状结构上递归调用方法并对结果进行汇总。

结构图

组合模式中的角色:

  • 抽象构件角色(Component):这是一个抽象角色,它给参加组合的对象定义出了公共的接口及默认行为,可以用来管理所有的子对象(在透明式的组合模式是这样的)。在安全式的组合模式里,构件角色并不定义出管理子对象的方法,这一定义由树枝结构对象给出。
  • 树叶构件角色(Leaf):树叶对象是没有下级子对象的对象,定义出参加组合的原始对象的行为。(原始对象的行为可以理解为没有容器对象管理子对象的方法,或者原始对象行为+管理子对象的行为(Add,Remove等)=面对客户代码的接口行为集合)
  • 树枝构件角色(Composite):代表参加组合的有下级子对象的对象,树枝对象给出所有管理子对象的方法实现,如Add、Remove等。组合模式实现的最关键的地方是--简单对象和复合对象必须实现相同的接口。这就是组合模式能够将组合对象和简单对象进行一致处理的原因。

实现

组合模式实现的最关键的地方是——简单对象和复合对象必须实现相同的接口。这就是组合模式能够将组合对象和简单对象进行一致处理的原因。

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Threading.Tasks;
  6.  
  7. namespace Composite
  8. {
  9. class Program
  10. {
  11. static void Main(string[] args)
  12. {
  13. // 创建根节点
  14. Composite root = new Composite("root");
  15. root.add(new Leaf("Leaf A"));
  16. root.add(new Leaf("Leaf B"));
  17.  
  18. // 创建第二层节点
  19. Composite branch = new Composite("branch");
  20. branch.add(new Leaf("branch BX"));
  21. branch.add(new Leaf("branch BY"));
  22. root.add(branch);
  23.  
  24. // 创建第三层节点
  25. Composite branch2 = new Composite("branch2");
  26. branch2.add(new Leaf("branch2 BBX"));
  27. branch2.add(new Leaf("branch2 BBY"));
  28. root.add(branch2);
  29.  
  30. // 叶子节点操作
  31. Composite branch3 = new Composite("branch3");
  32. Leaf leaf = new Leaf("Leaf L");
  33. Leaf leaf1 = new Leaf("Leaf L1");
  34. leaf.add(leaf1);
  35. leaf.delete(leaf1);
  36. branch3.add(leaf);
  37. branch3.add(leaf1);
  38. branch3.delete(leaf);
  39. root.add(branch3);
  40.  
  41. // 显示
  42. root.show(1);
  43.  
  44. Console.Read();
  45. }
  46. }
  47.  
  48. /// <summary>
  49. /// 抽象构件
  50. /// </summary>
  51. public abstract class Component
  52. {
  53. public string Name { get; set; }
  54. public Component(string name)
  55. {
  56. this.Name = name;
  57. }
  58.  
  59. // 添加一个叶子构件或树枝构件
  60. public abstract void add(Component component);
  61. // 删除一个叶子构件或树枝构件
  62. public abstract void delete(Component component);
  63. // 获取分支下的所有叶子构件和树枝构件
  64. public abstract void show(int depth);
  65. }
  66.  
  67. /// <summary>
  68. /// 叶子构件
  69. /// </summary>
  70. public class Leaf : Component
  71. {
  72. public Leaf(string name):base(name)
  73. { }
  74.  
  75. // 如果是叶子节点,则不允许进行添加节点,因为叶子节点下再没有节点了
  76. public override void add(Component component)
  77. {
  78. Console.WriteLine("叶子节点不能添加其他内容");
  79. }
  80.  
  81. // 如果是叶子节点,则不允许进行删除节点,因为叶子节点下再没有节点了
  82. public override void delete(Component component)
  83. {
  84. Console.WriteLine("叶子节点不能删除内容");
  85. }
  86.  
  87. public override void show(int depth)
  88. {
  89. // 输出叶子节点
  90. for (int i = 0; i < depth; i++)
  91. {
  92. Console.Write("-");
  93. }
  94. Console.WriteLine(this.Name);
  95. }
  96. }
  97.  
  98. /// <summary>
  99. /// 树构件
  100. /// </summary>
  101. public class Composite : Component
  102. {
  103. protected List<Component> _children = new List<Component>();
  104. public Composite(string name) : base(name)
  105. { }
  106.  
  107. public override void add(Component component)
  108. {
  109. _children.Add(component);
  110. }
  111.  
  112. public override void delete(Component component)
  113. {
  114. _children.Remove(component);
  115. }
  116.  
  117. public override void show(int depth)
  118. {
  119. // 输出树形结构层次
  120. for (int i=0; i<depth; i++)
  121. {
  122. Console.Write("-");
  123. }
  124. Console.WriteLine(this.Name);
  125.  
  126. // 向下遍历
  127. foreach (Component compontent in _children)
  128. {
  129. compontent.show(depth + 1);
  130. }
  131. }
  132. }
  133. }

运行后结果:

  1. 叶子节点不能添加其他内容
  2. 叶子节点不能删除内容
  3. -root
  4. --Leaf A
  5. --Leaf B
  6. --branch
  7. ---branch BX
  8. ---branch BY
  9. --branch2
  10. ---branch2 BBX
  11. ---branch2 BBY
  12. --branch3
  13. ---Leaf L1

使用场景

  • 需要表示一个对象整体或部分的层次结构。
  • 希望用户忽略组合对象与单个对象的不同,用户将统一地使用组合结构中的所有对象。

优缺点

优点:

  • 组合模式使得客户端代码可以一致地处理对象和对象容器,无需关系处理的单个对象,还是组合的对象容器。
  • 将”客户代码与复杂的对象容器结构“解耦。
  • 可以更容易地往组合对象中加入新的构件。

缺点:

  • 使得设计更加复杂。客户端需要花更多时间理清类之间的层次关系。(这个是几乎所有设计模式所面临的问题)。

C#设计模式-组合模式(Composite Pattern)的更多相关文章

  1. 浅谈设计模式--组合模式(Composite Pattern)

    组合模式(Composite Pattern) 组合模式,有时候又叫部分-整体结构(part-whole hierarchy),使得用户对单个对象和对一组对象的使用具有一致性.简单来说,就是可以像使用 ...

  2. 设计模式 - 组合模式(composite pattern) 迭代器(iterator) 具体解释

    组合模式(composite pattern) 迭代器(iterator) 具体解释 本文地址: http://blog.csdn.net/caroline_wendy 參考组合模式(composit ...

  3. 设计模式 -- 组合模式 (Composite Pattern)

    定义: 对象组合成部分整体结构,单个对象和组合对象具有一致性. 看了下大概结构就是集团总公司和子公司那种层级结构. 角色介绍: Component :抽象根节点:其实相当去总公司,抽象子类共有的方法: ...

  4. C#设计模式——组合模式(Composite Pattern)

    一.概述 在软件开发中,我们往往会遇上类似树形结构的对象体系.即某一对象既可能在树形结构中作为叶节点存在,也可能作为分支节点存在.比如在文件系统中,文件是作为叶节点存在,而文件夹就是分支节点.在设计这 ...

  5. 乐在其中设计模式(C#) - 组合模式(Composite Pattern)

    原文:乐在其中设计模式(C#) - 组合模式(Composite Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 组合模式(Composite Pattern) 作者:weba ...

  6. 设计模式系列之组合模式(Composite Pattern)——树形结构的处理

    说明:设计模式系列文章是读刘伟所著<设计模式的艺术之道(软件开发人员内功修炼之道)>一书的阅读笔记.个人感觉这本书讲的不错,有兴趣推荐读一读.详细内容也可以看看此书作者的博客https:/ ...

  7. 二十四种设计模式:组合模式(Composite Pattern)

    组合模式(Composite Pattern) 介绍将对象组合成树形结构以表示"部分-整体"的层次结构.它使得客户对单个对象和复合对象的使用具有一致性.示例有一个Message实体 ...

  8. 【设计模式】组合模式 Composite Pattern

    树形结构是软件行业很常见的一种结构,几乎随处可见,  比如: HTML 页面中的DOM,产品的分类,通常一些应用或网站的菜单,Windows Form 中的控件继承关系,Android中的View继承 ...

  9. python 设计模式之组合模式Composite Pattern

    #引入一 文件夹对我们来说很熟悉,文件夹里面可以包含文件夹,也可以包含文件. 那么文件夹是个容器,文件夹里面的文件夹也是个容器,文件夹里面的文件是对象. 这是一个树形结构 咱们生活工作中常用的一种结构 ...

  10. 设计模式-12组合模式(Composite Pattern)

    1.模式动机 很多时候会存在"部分-整体"的关系,例如:大学中的部门与学院.总公司中的部门与分公司.学习用品中的书与书包.在软件开发中也是这样,例如,文件系统中的文件与文件夹.窗体 ...

随机推荐

  1. node转发请求 .csv格式文件下载 中文乱码问题 + 文件上传笔记

    用户无法直接访问后台接口 需要node端转发请求 并将数据以.csv文件格式生成以供客户端下载. 很不幸出现了中文乱码的问题 挖了各种坟帖,下了各种依赖包,csv.json2csv.bufferHel ...

  2. jquery播放图片

    * { margin:0; padding:0; word-break:break-all; } body { background:#FFF; color:#333; font:12px/1.5em ...

  3. 【QT】QtConcurrent::run()+QThreadPool实现多线程

    往期链接: <QThread源码浅析> <子类化QThread实现多线程> <子类化QObject+moveToThread实现多线程> <继承QRunnab ...

  4. ubuntu18.04下stlink的一种安装方法

    安装前准备: 从软件包存储库中安装以下软件包: git gcc或clang或mingw32-gcc或mingw64-gcc(C编译器:很可能已经存在gcc) build-essential (在基于D ...

  5. Ques1,debug模式打不开的原因

    我尝试过一下三种方法打开pycharm环境下Flask的debug模式,但是都失败了.这三种具体方法如下: 1, 2, 3, 可以说上述方法都是常规操作,可是为啥还是不能打开debug模式呢? 因为: ...

  6. CSS之calc()

    calc() 函数支持任意CSS长度单位的混合计算,遵循标准数学运算优先级规则,可以动态计算长度值.注意,calc()函数内部的运算符两侧各加一个空白符,否则会产生解析错误. calc()使用的难点在 ...

  7. [MIT6.006] 20. Daynamic Programming II: Text Justification, Blackjack 动态规划II:文本对齐,黑杰克

    这节课通过讲解动态规划在文本对齐(Text Justification)和黑杰克(Blackjack)上的求解过程,来帮助我们理解动态规划的通用求解的五个步骤: 动态规划求解的五个"简单&q ...

  8. vue-count-to(简单好用的一个数字滚动插件)

    vue-count-to是一个无依赖,轻量级的vue组件,可覆盖easingFn. 1. 你可以设置两个属性startVal和endVal,它会自动判断计数或倒计时.支持vue-ssr.vue-cou ...

  9. Git仓库的提交记录乱成一团,怎么办?

    大家好,今天和大家聊聊git当中一个非常好用的功能--区间选择,它可以帮我们处理看起来非常复杂的提交记录.从而帮助我们很快找到我们需要的内容. 如果大家有参与过多人协同的项目开发,比如十几个人甚至更多 ...

  10. Internet 网络协议族

    1.linux目前支持多种协议族,每个协议族用一个net_porto_family结构实例来表示,在初始化时,会调用sock_register()函数初始化注册到net_families[NPROTO ...