组合模式
模式动机
对于树形结构,当容器对象(如文件夹)的某一个方法被调用时,将遍历整个树形结构,寻找也包含这个方法的成员对象(可以是容器对象,也可以是叶子对象,如子文件夹和文件)并调用执行。(递归调用)
由于容器对象和叶子对象在功能上的区别,在使用这些对象的客户端代码中必须有区别地对待容器对象和叶子对象,而实际上大多数情况下客户端希望一致地处理它们,因为对于这些对象的区别对待将会使得程序非常复杂。
组合模式描述了如何将容器对象和叶子对象进行递归组合,使得用户在使用时无须对它们进行区分,可以一致地对待容器对象和叶子对象,这就是组合模式的模式动机。
模式定义
组合模式(Composite Pattern):组合多个对象形成树形结构以表示“整体-部分”的结构层次。组合模式对单个对象(即叶子对象)和组合对象(即容器对象)的使用具有一致性。
组合模式又可以称为 “整体-部分”(Part-Whole)模式,属于对象的结构模式,它将对象组织到树结构中,可以用来描述整体与部分的关系。
Composite Pattern: Compose objects into tree structures to represent part-whole hierarchies. Composite lets clients treat individual objects and compositions of objects uniformly.
模式结构
组合模式包含以下角色:
- Component:抽象构件
- Leaf:叶子构件
- Composite:容器构件
- Client:客户类
叶子节点继承了它不应该实现的方法怎么办?
为了给客户端提供一致的接口(Component),抽象类中有些方法时叶点不该实现的.解决方法是空方法或者用叶节点调用时抛出异常或错误提示.
组合模式能否实现迭代?
当然可以,使用栈即可.实际上递归就是系统调用栈隐式管理状态,所以递归基本都能改为迭代
模式分析
- 组合模式的关键是定义了一个抽象构件类,它既可以代表叶子,又可以代表容器,而客户端针对该抽象构件类进行编程,无需知道它到底代表的是叶子还是容器,可以对其进行统一处理
- 同时容器对象与抽象构件类之间还建立一个聚合关联关系,在容器对象中既可以叶子,也可以包含容器,以此实现递归组合,形成一个树形结构
典型实现
典型的抽象构件角色代码:
1
2
3
4
5
6
7
8
9
public abstract class Component {
public abstract void add(Component c);
public abstract void remove(Component c);
public abstract Component getChild(int i);
public abstract void operation();
}
典型的叶子构件角色代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
public class Leaf extends Component {
public void add(Component c) { //异常处理或错误提示
}
public void remove(Component c) { //异常处理或错误提示
}
public Component getChild(int i) { //异常处理或错误提示
}
public void operation() {
//实现代码
}
}
典型的容器构件角色代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public class Composite extends Component {
private ArrayList list = new ArrayList();
public void add(Component c) {
list.add(c);
}
public void remove(Component c) {
list.remove(c);
}
public Component getChild(int i) {
(Component) list.get(i);
}
public void operation() {
for (Object obj : list) {
((Component) obj).operation();
}
}
}
模式优缺点
组合模式优点
- 可以清楚地定义分层次的复杂对象,表示对象的全部或部分层次,使得增加新构件也更容易。
- 客户端调用简单,客户端可以一致的使用组合结构或其中单个对象
- 定义了包含叶子对象和容器对象的类层次结构,叶子对象可以被组合成更复杂的容器对象,而这个容器对象又可以被组合,这样不断递归下去,可以形成复杂的树形结构。
- 更容易在组合体内加入对象构件,客户端不必因为加入了新的对象构件而更改原有代码
组合模式缺点
- 使设计变得更加抽象,对象的业务规则如果很复杂,则实现组合模式具有很大挑战性,而且不是所有的方法与叶子对象子类都有关联。
- 增加新构件时可能会产生一些问题,很难对容器中的构件类型进行限制
模式适用环境
- 需要表示一个对象整体或部分层次,在具有整体和部分的层次结构中,希望通过一种方式忽略整体与部分的差异,可以一致地对待它们
- 让客户能够忽略不同对象层次的变化,客户端可以针对抽象构件编程,无须关心对象层次结构的细节
- 对象的结构是动态的并且复杂程度不一样,但客户需要一致地处理它们
模式扩展
- 透明组合模式:子类实现抽象节点的所有方法,调用时要注意安全
- 安全组合模式:抽象组件只包含共有方法
