分析Java設(shè)計(jì)模式之組合模式
一、概述
我們對(duì)于這個(gè)圖片肯定會(huì)非常熟悉,這兩幅圖片我們都可以看做是一個(gè)文件結(jié)構(gòu),對(duì)于這樣的結(jié)構(gòu)我們稱之為樹形結(jié)構(gòu)。在數(shù)據(jù)結(jié)構(gòu)中我們了解到可以通過調(diào)用某個(gè)方法來遍歷整個(gè)樹,當(dāng)我們找到某個(gè)葉子節(jié)點(diǎn)后,就可以對(duì)葉子節(jié)點(diǎn)進(jìn)行相關(guān)的操作。我們可以將這顆樹理解成一個(gè)大的容器,容器里面包含很多的成員對(duì)象,這些成員對(duì)象即可是容器對(duì)象也可以是葉子對(duì)象。但是由于容器對(duì)象和葉子對(duì)象在功能上面的區(qū)別,使得我們?cè)谑褂玫倪^程中必須要區(qū)分容器對(duì)象和葉子對(duì)象,但是這樣就會(huì)給客戶帶來不必要的麻煩,作為客戶而已,它始終希望能夠一致的對(duì)待容器對(duì)象和葉子對(duì)象。這就是組合模式的設(shè)計(jì)動(dòng)機(jī):組合模式定義了如何將容器對(duì)象和葉子對(duì)象進(jìn)行遞歸組合,使得客戶在使用的過程中無須進(jìn)行區(qū)分,可以對(duì)他們進(jìn)行一致的處理。
二、 模式定義
組合模式組合多個(gè)對(duì)象形成樹形結(jié)構(gòu)以表示“整體-部分”的結(jié)構(gòu)層次。
組合模式對(duì)單個(gè)對(duì)象(葉子對(duì)象)和組合對(duì)象(組合對(duì)象)具有一致性,它將對(duì)象組織到樹結(jié)構(gòu)中,可以用來描述整體與部分的關(guān)系。同時(shí)它也模糊了簡(jiǎn)單元素(葉子對(duì)象)和復(fù)雜元素(容器對(duì)象)的概念,使得客戶能夠像處理簡(jiǎn)單元素一樣來處理復(fù)雜元素,從而使客戶程序能夠與復(fù)雜元素的內(nèi)部結(jié)構(gòu)解耦。
上面的圖展示了計(jì)算機(jī)的文件系統(tǒng),文件系統(tǒng)由文件和目錄組成,目錄下面也可以包含文件或者目錄,計(jì)算機(jī)的文件系統(tǒng)是用遞歸結(jié)構(gòu)來進(jìn)行組織的,對(duì)于這樣的數(shù)據(jù)結(jié)構(gòu)是非常適用使用組合模式的。
在使用組合模式中需要注意一點(diǎn)也是組合模式最關(guān)鍵的地方:葉子對(duì)象和組合對(duì)象實(shí)現(xiàn)相同的接口。這就是組合模式能夠?qū)⑷~子節(jié)點(diǎn)和對(duì)象節(jié)點(diǎn)進(jìn)行一致處理的原因。
三、 模式結(jié)構(gòu)
組合模式主要包含如下幾個(gè)角色:
1.Component :組合中的對(duì)象聲明接口,在適當(dāng)?shù)那闆r下,實(shí)現(xiàn)所有類共有接口的默認(rèn)行為。聲明一個(gè)接口用于訪問和管理Component子部件。
2.Leaf:葉子對(duì)象。葉子結(jié)點(diǎn)沒有子結(jié)點(diǎn)。
3.Composite:容器對(duì)象,定義有枝節(jié)點(diǎn)行為,用來存儲(chǔ)子部件,在Component接口中實(shí)現(xiàn)與子部件有關(guān)操作,如增加(add)和刪除(remove)等。
從模式結(jié)構(gòu)中我們看出了葉子節(jié)點(diǎn)和容器對(duì)象都實(shí)現(xiàn)Component接口,這也是能夠?qū)⑷~子對(duì)象和容器對(duì)象一致對(duì)待的關(guān)鍵所在。
四、 模式實(shí)現(xiàn)
在文件系統(tǒng)中,可能存在很多種格式的文件,如果圖片,文本文件、視頻文件等等,這些不同的格式文件的瀏覽方式都不同,同時(shí)對(duì)文件夾的瀏覽就是對(duì)文件夾中文件的瀏覽,但是對(duì)于客戶而言都是瀏覽文件,兩者之間不存在什么差別,現(xiàn)在只用組合模式來模擬瀏覽文件。UML結(jié)構(gòu)圖:
首先是文件類:File.java
public abstract class File { String name; public File(String name){ this.name = name; } public String getName() { return name; } public void setName(String name) { this.name = name; } public abstract void display(); }
然后是文件夾類:Folder.java,該類包含對(duì)文件的增加、刪除和瀏覽三個(gè)方法
public class Folder extends File{ private List<File> files; public Folder(String name){ super(name); files = new ArrayList<File>(); } /** * 瀏覽文件夾中的文件 */ public void display() { for(File file : files){ file.display(); } } /** * @desc 向文件夾中添加文件 * @param file * @return void */ public void add(File file){ files.add(file); } /** * @desc 從文件夾中刪除文件 * @param file * @return void */ public void remove(File file){ files.remove(file); } }
然后是三個(gè)文件類:TextFile.java、ImageFile.java、VideoFile.java
TextFile.java
public class TextFile extends File{ public TextFile(String name) { super(name); } public void display() { System.out.println("這是文本文件,文件名:" + super.getName()); } }
ImageFile.java
public class ImagerFile extends File{ public ImagerFile(String name) { super(name); } public void display() { System.out.println("這是圖像文件,文件名:" + super.getName()); } }
VideoFile.java
public class VideoFile extends File{ public VideoFile(String name) { super(name); } public void display() { System.out.println("這是影像文件,文件名:" + super.getName()); } }
最后是客戶端
public class Client { public static void main(String[] args) { /** * 我們先建立一個(gè)這樣的文件系統(tǒng) * 總文件 * * a.txt b.jpg c文件夾 * c_1.text c_1.rmvb c_1.jpg * */ //總文件夾 Folder zwjj = new Folder("總文件夾"); //向總文件夾中放入三個(gè)文件:1.txt、2.jpg、1文件夾 TextFile aText= new TextFile("a.txt"); ImagerFile bImager = new ImagerFile("b.jpg"); Folder cFolder = new Folder("C文件夾"); zwjj.add(aText); zwjj.add(bImager); zwjj.add(cFolder); //向C文件夾中添加文件:c_1.txt、c_1.rmvb、c_1.jpg TextFile cText = new TextFile("c_1.txt"); ImagerFile cImage = new ImagerFile("c_1.jpg"); VideoFile cVideo = new VideoFile("c_1.rmvb"); cFolder.add(cText); cFolder.add(cImage); cFolder.add(cVideo); //遍歷C文件夾 cFolder.display(); //將c_1.txt刪除 cFolder.remove(cText); System.out.println("-----------------------"); cFolder.display(); } }
運(yùn)行結(jié)果
五、 模式優(yōu)缺點(diǎn)
5.1、優(yōu)點(diǎn)
1、可以清楚地定義分層次的復(fù)雜對(duì)象,表示對(duì)象的全部或部分層次,使得增加新構(gòu)件也更容易。
2、客戶端調(diào)用簡(jiǎn)單,客戶端可以一致的使用組合結(jié)構(gòu)或其中單個(gè)對(duì)象。
3、定義了包含葉子對(duì)象和容器對(duì)象的類層次結(jié)構(gòu),葉子對(duì)象可以被組合成更復(fù)雜的容器對(duì)象,而這個(gè)容器對(duì)象又可以被組合,這樣不斷遞歸下去,可以形成復(fù)雜的樹形結(jié)構(gòu)。
4、更容易在組合體內(nèi)加入對(duì)象構(gòu)件,客戶端不必因?yàn)榧尤肓诵碌膶?duì)象構(gòu)件而更改原有代碼。
5.2、缺點(diǎn)
1、使設(shè)計(jì)變得更加抽象,對(duì)象的業(yè)務(wù)規(guī)則如果很復(fù)雜,則實(shí)現(xiàn)組合模式具有很大挑戰(zhàn)性,而且不是所有的方法都與葉子對(duì)象子類都有關(guān)聯(lián)
六、 模式適用場(chǎng)景
1、需要表示一個(gè)對(duì)象整體或部分層次,在具有整體和部分的層次結(jié)構(gòu)中,希望通過一種方式忽略整體與部分的差異,可以一致地對(duì)待它們。
2、讓客戶能夠忽略不同對(duì)象層次的變化,客戶端可以針對(duì)抽象構(gòu)件編程,無須關(guān)心對(duì)象層次結(jié)構(gòu)的細(xì)節(jié)。
七、 模式總結(jié)
1、 組合模式用于將多個(gè)對(duì)象組合成樹形結(jié)構(gòu)以表示“整體-部分”的結(jié)構(gòu)層次。組合模式對(duì)單個(gè)對(duì)象(葉子對(duì)象)和組合對(duì)象(容器對(duì)象)的使用具有一致性。
2、 組合對(duì)象的關(guān)鍵在于它定義了一個(gè)抽象構(gòu)建類,它既可表示葉子對(duì)象,也可表示容器對(duì)象,客戶僅僅需要針對(duì)這個(gè)抽象構(gòu)建進(jìn)行編程,無須知道他是葉子對(duì)象還是容器對(duì)象,都是一致對(duì)待。
3、 組合模式雖然能夠非常好地處理層次結(jié)構(gòu),也使得客戶端程序變得簡(jiǎn)單,但是它也使得設(shè)計(jì)變得更加抽象,而且也很難對(duì)容器中的構(gòu)件類型進(jìn)行限制,這會(huì)導(dǎo)致在增加新的構(gòu)件時(shí)會(huì)產(chǎn)生一些問題。
以上就是分析Java設(shè)計(jì)模式之組合模式的詳細(xì)內(nèi)容,更多關(guān)于Java 設(shè)計(jì)模式 組合模式的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Spring boot攔截器實(shí)現(xiàn)IP黑名單的完整步驟
這篇文章主要給大家介紹了關(guān)于Spring boot攔截器實(shí)現(xiàn)IP黑名單的完整步驟,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Spring boot攔截器具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06java實(shí)現(xiàn)學(xué)生選課系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)學(xué)生選課系統(tǒng),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-02-02Netty分布式NioEventLoop優(yōu)化selector源碼解析
這篇文章主要介紹了Netty分布式NioEventLoop優(yōu)化selector源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-03-03SpringBoot整合JPA數(shù)據(jù)源方法及配置解析
這篇文章主要介紹了SpringBoot整合JPA數(shù)據(jù)源方法及配置解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-08-08Java Quartz觸發(fā)器CronTriggerBean配置用法詳解
這篇文章主要介紹了Java Quartz觸發(fā)器CronTriggerBean配置用法詳解,本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08基于Java編寫一個(gè)PDF與Word文件轉(zhuǎn)換工具
前段時(shí)間一直使用到word文檔轉(zhuǎn)pdf或者pdf轉(zhuǎn)word,尋思著用Java應(yīng)該是可以實(shí)現(xiàn)的,于是花了點(diǎn)時(shí)間寫了個(gè)文件轉(zhuǎn)換工具,感興趣的可以了解一下2023-01-01Java中實(shí)現(xiàn)Comparator接口和用法實(shí)例(簡(jiǎn)明易懂)
這篇文章主要介紹了Java中實(shí)現(xiàn)Comparator接口和用法實(shí)例(簡(jiǎn)明易懂),本文給出實(shí)現(xiàn)Comparator接口的實(shí)例和使用這個(gè)接口的代碼實(shí)例,需要的朋友可以參考下2015-05-05