组合模式(Composite Pattern)
定义:
又叫整体-部分(Part-Whole)模式,是一种将对象组合成树状的层次结构的模式,用来表示“整体-部分”的关系,使用户对单个对象和组合对象具有一致的访问性,属于结构型设计模式。
组合模式一般用来描述整体与部分的关系,将对象组织到树形结构中,顶层的节点被称为根节点,根节点下面可以包含树枝节点和叶子节点,树枝节点下面又可以包含树枝节点和叶子节点。
根节点和树枝节点本质上属于同一种数据类型,可以作为容器使用,而叶子节点与树枝节点在语义上不属于同一种类型。但是在组合模式中,会把树枝节点和叶子节点看作属于同一种类型,让他们具备一致行为。
在组合模式中,整个树形结构中的对象都属于同一种类型,带来的好处就是用户不需要辨别是树枝节点还是叶子节点,可以直接进行操作,给用户的使用带来极大的便利。
优点:
- 组合模式使得客户端代码可以一致的处理单个对象和组合对象,不用关心自己处理的是单个对象还是组合对象,简化了客户端代码。
- 更容易在组合体内加入新的对象,客户端不会因为加入了新对象而更改源代码,满足“开闭原则”。
缺点:
- 设计复杂,客户端需要花更多时间来理清类之间的层次关系。
- 不容易限制容器中的构件。
- 不容易用继承的方法来增加构件的新功能。
结构:
- 抽象构件(Component)角色:为树叶构件声明公共接口,并实现他们的默认行为。在透明式的组合模式中抽象构件还声明访问和管理子类的接口。在安全式的组合模式中不声明访问和管理子类的接口,管理工作由树枝构件完成。(总的抽象类或接口,定义一些通用方法,如新增、删除)。
- 树叶构件(Leaf)角色:是组合中的叶节点对象,没有子节点,由于继承或实现抽象构件。
- 树枝构件(Composite)角色/中间构件:是组合中的分支节点对象,有子节点,用于继承和实现抽象构件。主要作用是存储和管理子部件,通常包含Add()、Remove()、GetChild()等方法。
组合模式分为透明式的组合模式和安全式的组合模式。
- 透明模式
在该方式中,由于抽象构件声明了所有子类中的全部方法,所有客户端不需要区别树叶对象和树枝对象,对客户端来说是透明的。缺点是:树叶构件本来没有Add()、Remove()、GetChild()方法,却要实现他们(空实现或者抛异常),这样会有安全性问题。
- 安全方式
在该方式中,将管理子构件的方法移到树枝构件中,抽象构件和树叶构件没有对子对象的管理方法,就避免了上一种方式的安全性问题,但由于叶子和分支有不同的接口,客户端在调用时要知道树叶对象和树枝对象的存在,所以失去了透明性。
实现:
应用场景:
- 在需要表示一个对象整体与部分的层次结构的场合。
- 要求对用户隐藏组合对象与单个对象的不同,用户可以用统一的接口使用组合结构中的所有对象的场合。
扩展:
如果对前面介绍的组合模式中的树叶节点和树枝节点进行抽象,也就是说树叶节点和树枝节点还有子节点,这时组合模式就扩展成复杂的组合模式了,如 Java AWT/Swing 中的简单组件 JTextComponent 有子类 JTextField、JTextArea,容器组件 Container 也有子类 Window、Panel。