1. 背景
在现实生活中,存在很多“部分-整体”的关系,例如,大学中的部门与学院、总公司中的部门与分公司、学习用品中的书与书包、生活用品中的衣月艮与衣柜以及厨房中的锅碗瓢盆等。
在软件开发中也是这样,例如,文件系统中的文件与文件夹、窗体程序中的简单控件与容器控件等。对这些简单对象与复合对象的处理,如果用组合模式来实现会很方便
在Windows系统中,文件系统是一个树结构,树上长有节点。树的节点有两种,一种是树枝节点,也就是目录,有内部树结构,在图中涂有颜色;另一种是树叶节点,也就是文件,没有内部树结构;
假如现在的需求是模拟上述的文件系统,并且对目录进行遍历显示其结构
在还没有涉及到组合模式的时候我们会这样处理
文件夹包含文件,同时文件夹也可以包含文件夹
File
public class File {
private String name;
public String getName() {
return name;
}
public File(String name) {
this.name = name;
}
}
Fold
public class Folder {
private List<Folder> folderList;
private List<File> fileList;
private String name;
public String getName() {
return name;
}
public Folder(String name) {
this.name = name;
fileList = new ArrayList<File>();
folderList = new ArrayList<Folder>();
}
public void add(Folder f){
folderList.add(f);
}
public void add(File f){
fileList.add(f);
}
/**
*
* @param tab 分隔符表示缩进 为了显示层次结构
*/
public void traverse(String tab){//打印文件结构
System.out.println(tab+"+"+getName());
for (Folder folder : folderList) {
folder.traverse(tab+" ");
}
for (File file : fileList) {
System.out.println(tab+"-"+file.getName());
}
}
}
模拟遍历
public static void main(String[] args) {
Folder f1 = new Folder("Folder A");
Folder f2 = new Folder("Folder B");
f1.add(f2);
f1.add(new File("File1"));
f1.add(new File("File2"));
f2.add(new File("File3"));
f2.add(new File("File4"));
f1.traverse("");
}
上面可以实现我们的要求,但是我们必须要把File和Fold区别对待,必须要了解各个类的操作,没有统一的操作方式;如果我们在对树上的节点和叶子操作能提供一致性的方式而不考虑是叶子还是节点,就可以使用组合方法
2. 定义与特点
定义:
组合(Composite)模式,有时又叫作部分-整体模式,它是一种将对象组合成树状的层次结构的模式,用来表示“部分-整体”的关系,组合模式使得用户对单个对象和组合对象的访问具有一致性,组合能让客户对单个对象和组合对象具有一致的访问性
优点:
-
组合模式使得客户端代码可以一致地处理单个对象和组合对象,无须关心自己处理的是单个对象,还是组合对象,这简化了客户端代码;
-
更容易在组合体内加入新的对象,客户端不会因为加入了新的对象而更改源代码,满足开闭原则
缺点:
-
设计较复杂,客户端需要花更多时间理清类之间的层次关系;
-
不容易限制容器中的构件;
-
不容易用继承的方法来增加构件的新功能
应用场景:
-
在需要表示一个对象整体与部分的层次结构的场合
-
要求对用户隐藏组合对象与单个对象的不同,用户可以用统一的接口使用组合结构中的所有对象的场合
① 透明式的组合模式
在该方式中,抽象构件声明了所有子类中的全部方法,所以客户端无须区别树叶对象和树枝对象,对客户端来说是透明的。但其缺点是:树叶构件本来没有 Add()、Remove() 及 GetChild()
方法,却要实现它们(空实现或抛异常),这样会带来一些安全性问题
抽象构件(Component)角色: 它的主要作用是为树叶构件和树枝构件声明公共接口,并实现它们的默认行为。在透明式的组合模式中抽象构件还声明访问和管理子类的接口;在安全式的组合模式中不声明访问和管理子类的接口,管理工作由树枝构件完成
树叶构件(Leaf)角色: 是组合中的叶节点对象,它没有子节点,用于实现抽象构件角色中 声明的公共接口
树枝构件(Composite)角色: 是组合中的分支节点对象,它有子节点。它实现了抽象构件角色中声明的接口,它的主要作用是存储和管理子部件,通常包含 Add()、Remove()、GetChild()
等方法
② 安全式的组合模式
在该方式中,将管理子构件的方法移到树枝构件中,抽象构件和树叶构件没有对子对象的管理方法,这样就避免了上一种方式的安全性问题,但由于叶子和分支有不同的接口,客户端在调用时要知道树叶对象和树枝对象的存在,所以失去了透明性