1.組合模式介紹
? ? ? ? 組合模式(Composite Pattern)也稱為部分整體模式艺智,是七大結(jié)構(gòu)型設(shè)計模式之一。
? ? ? ? 組合模式將一組相似的對象看作一個對象處理褐筛,并根據(jù)一個樹狀結(jié)構(gòu)來組合對象,然后提供一個統(tǒng)一的方法去訪問相應(yīng)的對象叙身。如下圖就很形象了渔扎,這類似于數(shù)據(jù)結(jié)構(gòu)中的樹結(jié)構(gòu)就是組合模式。
2.組合模式的定義
? ? ? ? 將對象組合成樹形結(jié)構(gòu)以表示“部分-整體”的層次結(jié)構(gòu)信轿,使得用戶對單個對象和組合對象的使用具有一致性晃痴。
3.組合模式的使用場景
? ? ? ? 表示對象的部分-整體層次結(jié)構(gòu)時。
? ? ? ? 從一個整體中能夠獨立出部分模塊或功能的場景财忽。
4.組合模式的UML類圖
5.角色介紹
? ? ? ? Component:抽象根節(jié)點倘核,為組合模式中的對象聲明接口,在適當?shù)那闆r下即彪,實現(xiàn)所有類共有接口的缺省行為紧唱。聲明一個接口用于訪問和管理Component的子節(jié)點×バ#可以在遞歸結(jié)構(gòu)中定義一個接口漏益,用于訪問一個父節(jié)點,并在合適的情況下實現(xiàn)它深胳。
? ? ? ? Composite:定義有子節(jié)點的那些枝干節(jié)點的行為绰疤,存儲子節(jié)點,在Component接口中實現(xiàn)與子節(jié)點有關(guān)的操作舞终。
? ? ? ? Leaf:在組合中表示葉子節(jié)點對象轻庆,葉子節(jié)點沒有子節(jié)點癣猾,在組合模式中定義節(jié)點對象的行為。
? ? ? ? Client:客戶端調(diào)用余爆,通過Component接口操縱組合節(jié)點的對象纷宇。
6.模板
7.例子
例子分析
? ? ? ? 例子是透明組合模式呐粘,文件夾是枝干節(jié)點,文件是葉子節(jié)點转捕。文件類相對于文件夾類來說既不支持添加也不支持刪除作岖,它是文件系統(tǒng)中最小的分割單位,我們實現(xiàn)邏輯可以拋出一個不支持異常即可(例子中以打印日志說明不支持該操作為以及不做處理了)五芝。
8.Android源碼中的組合模式
? ? ? ? Android源碼中View和ViewGroup的嵌套組合是非常經(jīng)典的組合模式痘儡。只有ViewGroup才能包含其他的View,比如LinearLayout能包含TextView枢步,而TextView不能包含LinearLayout的沉删。TextView直接繼承于View,TextVie相當于具體的葉子節(jié)點醉途,而LinearLayout相當于具體的枝干節(jié)點矾瑰。ViewGroup相對于View只是多了幾個對視圖操作的方法。
9.總結(jié)
? ? ? ? 組合模式和解釋器模式有一定的相似隘擎,它們在迭代對象時都涉及遞歸的調(diào)用殴穴,組合模式所提供的屬性層次結(jié)構(gòu)使得我們能夠一視同仁對待單個對象和對象集合,不過是以犧牲類的單一原則換來的货葬,而且組合模式是通過繼承來實現(xiàn)的采幌,缺少彈性。
? ? ? ? 優(yōu)點:1.組合模式可以清楚地定義分層次的復雜對象震桶,表示對象的全部或部分層次休傍,它讓高層模塊忽略了層次的差異,方便對整個層次結(jié)構(gòu)進行控制蹲姐。
? ? ? ? 2.高層模塊可以一致地使用一個組合結(jié)構(gòu)或其追蹤單個對象磨取,不必關(guān)心處理的是單個對象還是整個組合結(jié)構(gòu),簡化了高層模塊的代碼淤堵。
? ? ? ? 3.在組合模式中增加新的枝干構(gòu)件和葉子構(gòu)件都很方便寝衫,無需對現(xiàn)有類庫進行任何修改,符合“開閉原則”拐邪。
? ? ? ? 4.組合模式為樹形結(jié)構(gòu)的面向?qū)ο髮崿F(xiàn)提供了一種靈活的解決方案慰毅,通過葉子對象和枝干對象的遞歸組合,可以形成復雜的樹形結(jié)構(gòu)扎阶,但對樹形結(jié)構(gòu)的控制非常簡單汹胃。
? ? ? ? 缺點:在新增構(gòu)件時不好對枝干中的構(gòu)件類型進行限制婶芭,不能依賴類型系統(tǒng)來施加這些約束,因為大多數(shù)情況下着饥,它們都是來自于相同的抽象層犀农。
PS:讀《Android源碼設(shè)計模式解析與實戰(zhàn)》中的組合模式之后的一些筆記與感悟