设计模式 组合模式(Composite Pattern)
组合模式简绍
组合模式(Composite Pattern)是一种结构型设计模式,它允许你将对象组合成树形结构来表示“部分-整体”的层次结构。组合模式使得客户端可以用一致的方式处理单个对象和组合对象。这样,可以在不知道对象具体类型的条件下操作对象集合。
组合模式主要包含三种角色:
- Component(组件):声明一个接口,在适当情况下,此接口被叶节点和容器节点共享。
- Leaf(叶节点):定义终端对象。
- Composite(容器节点):定义具有子部件的那些部件的行为。存储子部件。 组合模式的应用场景
组合模式的优缺点
组合模式的优点
- 一致性:客户端可以一致地处理单个对象和组合对象。
- 灵活性:可以很容易地添加新的叶子节点或组合节点。
- 透明性:客户端不需要知道对象的具体类型就可以操作对象。
组合模式的缺点
- 复杂性:对于简单操作,组合模式可能会引入不必要的复杂性。
- 额外开销:组合模式可能会导致额外的内存和 CPU 开销,尤其是在处理大量对象时
UML图
定义一个公共接口或抽象类,它声明了所有叶子节点和容器节点共有的操作。
package CompositePatternModel;
public interface Component {
void test();
Component getChild(int i);
public void operation();
public void add(Component component);
}
实现 Component 接口中定义的操作,但不关心其他与子节点相关的操作。
package CompositePatternModel;
import java.util.ArrayList;
import java.util.List;
public class Composite implements Component{
private String name;
private List<Component> children = new ArrayList<>();
public Composite(String name){
this.name = name;
}
@Override
public void test() {
}
@Override
public Component getChild(int i) {
return children.get(i);
}
@Override
public void operation() {
System.out.println("Composite " + name + " operation");
for (Component c : children) {
c.operation();
}
}
@Override
public void add(Component component) {
children.add(component);
}
}
实现 Component 接口中定义的操作,并且实现与子节点相关的操作。客户端可以一致地对待单个对象和组合对象。
package CompositePatternModel;
public class Pattern implements Component{
private String name;
public Pattern(String name){
this.name = name;
}
@Override
public void test() {
System.out.println("test");
}
@Override
public Component getChild(int i) {
System.out.println("getChild");
return null;
}
@Override
public void operation() {
System.out.println("operation: d" + name);
}
@Override
public void add(Component component) {
System.out.println("sdf"+ component);
}
}
客户端可以一致地对待单个对象和组合对象。
package CompositePatternModel;
public class Main {
public static void main(String[] args) {
Pattern pattern = new Pattern("pattern1");
Pattern pattern1 = new Pattern("pattern2");
Composite composite = new Composite("Comp");
composite.add(pattern);
composite.add(pattern1);
composite.operation();
}
}
-
表示部分-整体层次结构
当你需要表示一个具有层次结构的对象集合时,组合模式是非常有用的。这种结构通常包含多个层次,每个层次上的对象可以是一个独立的实体(叶节点),也可以是一个包含其他对象的容器(组合节点)。
示例:- 文件系统中的目录和文件:目录可以包含其他目录和文件。
- 组织结构图:部门可以包含其他部门或员工。
- UI控件:一个复合控件可以包含其他控件(如面板包含按钮、文本框等)
-
透明地处理单个对象和组合对象
当客户端代码需要以一致的方式处理单个对象和组合对象时,组合模式可以让客户端无需关心处理的是单个对象还是组合对象。
示例:- 图形编辑器:图形可以是单独的线条、矩形等基本图形,也可以是由多个基本图形组成的复合图形。
- 渲染引擎:渲染对象可以是单一的几何体,也可以是由多个几何体组成的复杂场景。
-
动态地增删对象
当需要在运行时动态地添加或移除对象,并且这些对象可能是单个实体也可能是包含其他对象的容器时,组合模式可以很好地支持这种需求。示例:
游戏中的场景管理:游戏场景可以包含单个游戏对象,也可以包含其他场景。
UI 界面构建:一个窗口可以包含多个面板,每个面板又可以包含多个组件。 -
需要递归处理的对象集合
当需要对一个层次结构进行递归处理时,组合模式可以简化递归逻辑。
示例:- 企业组织结构的遍历:遍历整个公司的部门和员工。
- 文件系统的遍历:递归地遍历目录和文件。
-
需要一致的接口
当希望在不同类型的对象之间提供一致的接口时,组合模式可以确保所有的对象(无论是叶子还是组合节点)都遵循相同的接口定义。
示例:
操作系统中的文件和目录:文件和目录都需要支持读写、删除等操作。
网页中的 DOM 结构:DOM 节点可以是文本节点也可以是包含其他节点的元素节点。
组合模式提供了一种优雅的方式来组织和操作层次结构中的对象,使得客户端可以在不知道对象具体类型的情况下操作对象集合。这种模式在很多领域都有广泛的应用,特别是在需要表示层次关系的场景中。