组合模式(Composite)属于结构性模式,它描述了对象间的组合关系。
对象间常常通过树结构来组织(包含)起来,以实现整体-部分的层次结构。整体上可以看做是一个组合对象。
抛却各种复杂的术语,组合模式的特点是:
对象通过实现(继承)统一的接口(抽象类),调用者对单一对象和组合对象的操作具有一致性。
组合模式好处:
1.使客户端调用简单,客户端可以一致的使用组合结构或其中单个对象,用户就不必关系自己处理的是单个对象还是整个组合结构,这就简化了客户端代码。
2.更容易在组合体内加入对象部件. 客户端不必因为加入了新的对象部件而更改代码。
如何使用Composite?
首先定义一个接口或抽象类,这是设计模式通用方式了,其他设计模式对接口内部定义限制不多,Composite却有个规定,
那就是要在接口内部定义一个用于访问和管理Composite组合体的对象们(或称部件Component).
从结构图上来看组合模式和装饰模式很相近,相同之处是两种模式都提供了一种组合,装饰模式提供的是一种功能的组合,而组合模式提供的是一种对象组合。不同之处在于装饰模式只维护一个构件的引用,而组合模式中的组合对象提维护的是一个组件池,数量上有区别,这个区别就决定了装饰模式并不能构成树型结构,而组合模式则可以。 从业务逻辑上来讲,装饰模式的使用是为了给构件增加新的行为特征,而组合模式是为了整体-局部的层次型管理。从应用上来讲,组合模式的应用比装饰模式更具一般化,因此使用更广。
另外装饰模式中装饰者和构件并不一定要求同一类对象,在使用上也有区别,而组合模式中的组合对象和叶子对象都是同一类对象,在使用上并无区别。
组合模式的设计方法不仅在程序设计上应用非常广,在业务设计上,数据结构设计上都应用非常广泛。
在使用组合模式时,为了提供更灵活的访问,在接口组件提供对父节点的访问接口,叶子对象和组合对象都会实现这种接口,比如Delphi中Wincontrol类中的Parent属性,C#中Control类中的Parent等。还可以提供对象检索之类的功能。
在具体实现组合模式时,有时候根据需要,组合对象的子对象池只保持一个对子对象的引用,所有的对象会用一个集合来进行管理,这样做的好处主要是为了方便对对象的检索和管理。比如索引对象,释放资源。树型访问有的时候会比较慢。典型应用就是Delphi中的Form类。所有在Form中创建并指定了Owner属性的控件,都在Form的Components列表中。
类图如下: