组合模式,是由GoF提出的23种软件设计模式的一种。其定义为:将对象组合成树形结构以表示“整体—部分”的层次结构。Composite模式使单个对象和组合对象的使用具有一致性。
Composite模式是构造型的设计模式之一,通过递归手段来构造诸如文件系统之类的树形的对象结构;Composite模式所代表的数据构造是一群具有统一接口界面的对象集合,并可以通过一个对象来访问所有的对象(遍历)。
结构图如下所示:
用户使用Component类接口与组合结构中的对象就行交互。如果接受者是一个叶节点,则直接处理请求。如果接受者是一个Composite,它通常将请求发送给它的子部件,在转发请求之前或之后可能执行一些辅助操作。
举个例子:
应用Composite模式来实现文件系统的文件/目录结构:
Root:File与Folder的共通接口界面。相当于Component。
Folder:目录。目录下面有子目录,文件。相当于Composite。
File:文件。存在于目录之中。相当于Leaf。
Client类:测试类或者说使用类。
代码如下:
import java.util.*;
interface Root {
public boolean addFile(Root file);
public boolean removeFile(Root file);
public List<Root> getFile();
public void display();
}
class File implements Root {
String name;
public File(String name) {
this.name = name;
}
@Override
public List<Root> getFile() {
// TODO Auto-generated method stub
return null;
}
@Override
public boolean addFile(Root file) {
// TODO Auto-generated method stub
return false;
}
@Override
public boolean removeFile(Root file) {
// TODO Auto-generated method stub
return false;
}
@Override
public void display() {
// TODO Auto-generated method stub
System.out.println(" |___"+name);
}
}
class Folder implements Root {
String name;
List<Root> folder;
public Folder(String name) {
this.name = name;
this.folder = new ArrayList<Root>();
}
@Override
public List<Root> getFile() {
// TODO Auto-generated method stub
return folder;
}
@Override
public boolean addFile(Root file) {
// TODO Auto-generated method stub
return folder.add(file);
}
@Override
public boolean removeFile(Root file) {
// TODO Auto-generated method stub
return folder.remove(file);
}
@Override
public void display() {
// TODO Auto-generated method stub
System.out.println(name);
for (Root f : folder) {
if(f instanceof Folder){
System.out.print("|__");
}
f.display();
}
}
}
public class Client {
public static void main(String args[]) {
//构造一个文件目录
Root root1 = new Folder("c://");
Root root2 = new Folder("d://");
Root win = new Folder("windows");
Root sys = new Folder("system");
Root hw = new File("HelloWorld.java");
root1.addFile(win);
root1.addFile(sys);
win.addFile(hw);
root1.display();
root2.display();
}
}
输出为:
c:/
|__windows
|___HelloWorld.java
|__system
d:/
(恰好是一个树形文件结构)