??? 組合模式:將對(duì)象以樹(shù)形結(jié)構(gòu)組織起來(lái),以達(dá)成“部分-整體” 的層次結(jié)構(gòu),使得客戶端對(duì)單個(gè)對(duì)象和組合對(duì)象的使用具有一致性.
??? 適用環(huán)境:

u?????? 你想表示對(duì)象的部分-整體層次結(jié)構(gòu)。

u?????? 你希望用戶忽略組合對(duì)象與單個(gè)對(duì)象的不同,用戶將統(tǒng)一地使用組合結(jié)構(gòu)中的所有對(duì)象。

Composite好處
:
1.使客戶端調(diào)用簡(jiǎn)單,客戶端可以一致的使用組合結(jié)構(gòu)或其中單個(gè)對(duì)象,用戶就不必關(guān)系自己處理的是單個(gè)對(duì)象還是整個(gè)組合結(jié)構(gòu),這就簡(jiǎn)化了客戶端代碼。
2.更容易在組合體內(nèi)加入對(duì)象部件. 客戶端不必因?yàn)榧尤肓诵碌膶?duì)象部件而更改代碼。

組合模式的組成

1)???????? 抽象構(gòu)件角色Component:它為組合中的對(duì)象聲明接口,也可以為共有接口實(shí)現(xiàn)缺省行為。

2)?????? 樹(shù)葉構(gòu)件角色Leaf:在組合中表示葉節(jié)點(diǎn)對(duì)象——沒(méi)有子節(jié)點(diǎn),實(shí)現(xiàn)抽象構(gòu)件角色聲明的接口。

3)?????? 樹(shù)枝構(gòu)件角色Composite:在組合中表示分支節(jié)點(diǎn)對(duì)象——有子節(jié)點(diǎn),實(shí)現(xiàn)抽象構(gòu)件角色聲明的接口;存儲(chǔ)子部件。

下圖為組合模式的類圖表示。
comp1.bmp
三、安全性與透明性

組合模式中必須提供對(duì)子對(duì)象的管理方法,不然無(wú)法完成對(duì)子對(duì)象的添加刪除等等操作,也就失去了靈活性和擴(kuò)展性。但是管理方法是在Component中就聲明還是在Composite中聲明呢?

一種方式是在Component里面聲明所有的用來(lái)管理子類對(duì)象的方法,以達(dá)到Component接口的最大化(如下圖所示)。目的就是為了使客戶看來(lái)在接口層次上樹(shù)葉和分支沒(méi)有區(qū)別——透明性。但樹(shù)葉是不存在子類的,因此Component聲明的一些方法對(duì)于樹(shù)葉來(lái)說(shuō)是不適用的。這樣也就帶來(lái)了一些安全性問(wèn)題。
?? 另一種方式就是只在Composite里面聲明所有的用來(lái)管理子類對(duì)象的方法(如下圖所示)。這樣就避免了上一種方式的安全性問(wèn)題,但是由于葉子和分支有不同的接口,所以又失去了透明性。
??《設(shè)計(jì)模式》一書認(rèn)為:在這一模式中,相對(duì)于安全性,我們比較強(qiáng)調(diào)透明性。對(duì)于第一種方式中葉子節(jié)點(diǎn)內(nèi)不需要的方法可以使用空處理或者異常報(bào)告的方式來(lái)解決。
如何使用Composite?
首先定義一個(gè)接口或抽象類Component

package?composite;
import?java.util.Iterator;
public?abstract?class?Equipment{
????
//名稱
????private?String?name;
????
//價(jià)格
????public?abstract?double?netPrice();
????
//增加一個(gè)部件
????public?boolean?add(Equipment?equ){
????????
return?false;
????}

????
//刪除一個(gè)部件
????public?boolean?remove(Equipment?equ){
????????
return?false;
????}

????
//迭代指針
????public?Iterator?iter(){
????????
return?null;
????}

????
//構(gòu)造方法
????public?Equipment(String?name){
????????
this.name=name;
????}

}
Disk類Leaf角色
package?composite;
public?class?Disk?extends?Equipment{
????
//構(gòu)造方法
????public?Disk(String?name){
????????
super(name);
????}

????
//定價(jià)
????public?double?netPrice(){
????????
return?1.2;
????}

}
CompositeEquipment 類Composite角色
package?composite;
import?java.util.*;
public?abstract?class?CompositeEquipment??extends?Equipment{
????
private?List?equipment=new?ArrayList();
????
public?CompositeEquipment(String?name){
????????
super(name);
????}

????
public?boolean?add(Equipment?equ){
????????equipment.add(equ);
????????
return?true;
????}

????
public?boolean?remove(Equipment?equ){
????????equipment.remove(equ);
????????
return?true;
????}

????
????
public?Iterator?iter(){
????????
return?equipment.iterator();
????}

????
????
public?double?netPrice(){
????????
double?netprice=0;
????????Iterator?it
=equipment.iterator();
????????
while(it.hasNext()){
????????????netprice
+=((Equipment)it.next()).netPrice();
????????}

????????
return?netprice;
????}

????
}
CompositeEquipment的兩個(gè)具體類
package?composite;
public?class?Chassis?extends?CompositeEquipment{
????
public?Chassis(String?name){
????????
super(name);
????}

????
????
public?double?netPrice(){
????????
return?3.0+super.netPrice();
????}

????
}
package?composite;
public?class?Cabinet?extends?CompositeEquipment{
????
public?Cabinet(String?name){
????????
super(name);
????}

????
????
public?double?netPrice(){
????????
return?3.0+super.netPrice();
????}

????
}
運(yùn)行類
package?composite;
public?class?Composite{
????
public?static?void?main(String[]?args){
????????Cabinet?cabinet
=new?Cabinet("Tomer");
????????Chassis?chassis
=new?Chassis("PC?chassis");
????????
????????cabinet.add(chassis);
????????chassis.add(
new?Disk("120Gb"));
????????
????????System.out.println(cabinet.netPrice());
????}

}
運(yùn)行結(jié)果如:
7.2
Press any key to continue...


參考資料:http://blog.csdn.net/ai92/archive/2005/02/23/298336.aspx
?http://www.lvjiyong.com/books/DesignPatterns/21.html
http://www.jdon.com/designpatterns/composite.htm