组合模式
1、模式动机
- 对于树形结构,当容器对象(如文件夹)的某一个方法被调用时,将遍历整个树形结构,寻找也包含这个方法的成员对象(可以是容器对象,也可以是叶子对象,如子文件夹和文件)并调用执行。(递归调用)
- 由于容器对象和叶子对象在功能上的区别,在使用这些对象的客户端代码中必须有区别地对待容器对象和叶子对象,而实际上大多数情况下客户端希望一致地处理它们,因为对于这些对象的区别对待将会使得程序非常复杂。
- 组合模式描述了如何将容器对象和叶子对象进行递归组合,使得用户在使用时无须对它们进行区分,可以一致地对待容器对象和叶子对象,这就是组合模式的模式动机。
2、模式定义
- 组合模式(Composite Pattern):组合多个对象形成树形结构以表示“整体-部分”的结构层次。组合模式对单个对象(即叶子对象)和组合对象(即容器对象)的使用具有一致性。
- 组合模式又可以称为“整体-部分”(Part-Whole)模式,属于对象的结构模式,它将对象组织到树结构中,可以用来描述整体与部分的关系。
组合模式包含如下角色:
- Component: 抽象构件
- Leaf: 叶子构件
- Composite: 容器构件
- Client: 客户类
模式构件
3、模式分析
- 组合模式的关键是定义了一个抽象构件类,它既可以代表叶子,又可以代表容器,而客户端针对该抽象构件类进行编程,无须知道它到底表示的是叶子还是容器,可以对其进行统一处理。
- 同时容器对象与抽象构件类之间还建立一个聚合关联关系,在容器对象中既可以包含叶子,也可以包含容器,以此实现递归组合,形成一个树形结构。
- 文件系统组合模式结构图:
- 典型的抽象构件角色代码:
- 典型的叶子构件角色代码:
- 典型的容器构件角色代码:
4、模式案例与分析
- 实例一:水果盘
在水果盘(Plate)中有一些水果,如苹果(Apple)、香蕉(Banana)、梨子(Pear),当然大水果盘中还可以有小水果盘,现需要对盘中的水果进行遍历(吃),当然如果对一个水果盘执行“吃”方法,实际上就是吃其中的水果。使用组合模式模拟该场景。
package 组合模式2;
//叶子构件
public class Apple extends MyElement {
@Override
public void eat() {
System.out.println("吃苹果");
}
}
package 组合模式2;
//叶子构件
public class Banana extends MyElement {
@Override
public void eat() {
System.out.println("吃香蕉");
}
}
package 组合模式2;
//叶子构件
public class Pear extends MyElement {
@Override
public void eat() {
System.out.println("吃梨");
}
}
package 组合模式2;
// 抽象构件
public abstract class MyElement {
public abstract void eat();
}
package 组合模式2;
import java.util.ArrayList;
//容器
public class Plate extends MyElement {
private ArrayList arrayList = new ArrayList();
public void