组合模式
把部分和整体的关系用树形结构来表示,从而使客户端可以使用统一的方式处理部分对象和整体对象。
角色
抽象构件(Component):定义叶子和容器构件的共同点
叶子构件(Leaf):无子节点
容器构件(Composite):有容器特征,可以包含子节点
具体来说:容器构件的特征就是增加了add和remove子节点,以及进行索引的功能。
public interface Component {
void operation();
}
interface Leaf extends Component {}
interface Composite extends Component {
void add(Component c);
void remove(Component c);
Component getChild(int index);
}
举个例子:
一个文件杀毒系统,对于每一个不同类型的单个文件,属于叶子构件,因为不可能再有更细致的切分。 容器构件即文件夹,容器中可以包含容器,也可以包含叶子。
实现:
// 抽象构件,只有一个杀毒功能
public interface AbstractFile {
void killVirus(); //杀毒
}
// 定义三种类型的叶子节点,分别代表图像,文本,视频文件
class ImageFile implements AbstractFile {
private String name;
public ImageFile(String name) {
super();
this.name = name;
}
@Override
public void killVirus() {
System.out.println("---图像文件:" + name + ",进行查杀!");
}
}
class TextFile implements AbstractFile {
private String name;
public TextFile(String name) {
super();
this.name = name;
}
@Override
public void killVirus() {
System.out.println("---文本文件:" + name + ",进行查杀!");
}
}
class VideoFile implements AbstractFile {
private String name;
public VideoFile(String name) {
super();
this.name = name;
}
@Override
public void killVirus() {
System.out.println("---视频文件:" + name + ",进行查杀!");
}
}
//定义容器构件,文件夹,
//包含一个List用于存放其他实现AbstractFile类型的节点(文件夹,或者单个文件)
class Folder implements AbstractFile {
private String name;
// 定义容器,用来存放容器构件下的子节点
private List<AbstractFile> list = new ArrayList<AbstractFile>();
public Folder(String name) {
super();
this.name = name;
}
@Override
public void killVirus() {
System.out.println("---文件夹:" + name + ",进行查杀!");
// 如果有子文件夹的话,递归调用自己
for(AbstractFile file : list) {
file.killVirus();
}
}
public void add(AbstractFile file) {
list.add(file);
}
public void remove(AbstractFile file) {
list.remove(file);
}
public AbstractFile getChild(int index) {
return list.get(index);
客户端:无论是对文件还是对文件夹查杀,都可以直接调用killVirus方法。
public class Client {
public static void main(String[] args) {
AbstractFile f2,f3,f4,f5;
Folder f1 = new Folder("我的收藏");
f2 = new ImageFile("pic.jpg");
f3 = new TextFile("Hello.txt");
f1.add(f2);
f1.add(f3);
Folder f11 = new Folder("电影");
f4 = new VideoFile("笑傲江湖.avi");
f5 = new VideoFile("神雕侠侣.avi");
f11.add(f4);
f11.add(f5);
f1.add(f11);
f1.killVirus();
}
}
运行结果:
开发中的应用场景
操作系统的资源管理器
GUI中的容器层次图
XML文件解析
Junit单元测试框架