Composite(組合)--對象結構型模式

来源:http://www.cnblogs.com/mgp200866130/archive/2016/05/28/5538013.html
-Advertisement-
Play Games

1.意圖 將對象組合成樹形結構以表示“部分-整體”的層次結構。Composite使得用戶對單個對象和組合對象的使用具有一致性。 2.動機 可以組合多個簡單組件以形成一些較大的組件,這些組件又可以組合成更大的組件。Composite模式描述瞭如何使用遞歸組合,使得用戶不必對這些類進行區別。 3.適用性 ...


1.意圖

    將對象組合成樹形結構以表示“部分-整體”的層次結構。Composite使得用戶對單個對象和組合對象的使用具有一致性。

2.動機

    可以組合多個簡單組件以形成一些較大的組件,這些組件又可以組合成更大的組件。Composite模式描述瞭如何使用遞歸組合,使得用戶不必對這些類進行區別。

3.適用性

  • 表示對象的部分-整體層次結構。
  • 希望用戶忽略組合對象與單個對象的不同,用戶將統一地使用組合結構中的所有對象。

4.結構

    

    5.代碼實例

    

#include <memory>
#include <vector>

class Graphic
{
public:
    Graphic(std::shared_ptr<Graphic> pParent);
    virtual void Add(std::shared_ptr<Graphic>& pGraphic);
    virtual void Remove(std::shared_ptr<Graphic>& pGraphic);
    virtual std::shared_ptr<Graphic> GetChild(int iIndex);
    virtual std::shared_ptr<Graphic>& GetParent();
    virtual void Opereate();

protected:
    std::shared_ptr<Graphic> m_pParent;

    std::vector<std::shared_ptr<Graphic>> m_vecChildrenGraphics;

}; 

class Circle : public Graphic
{
public:
    Circle(std::shared_ptr<Graphic> pParent);
    void Opereate();
};


class Retangle : public Graphic
{
public:
    Retangle(std::shared_ptr<Graphic> pParent);
    void Opereate();
};


class Line : public Graphic
{
public:
    Line(std::shared_ptr<Graphic> pParent);
    void Opereate();
};

class Picture : public Graphic
{
public:
    Picture(std::shared_ptr<Graphic> pParent);
    virtual void Add(std::shared_ptr<Graphic>& pGraphic);
    virtual void Remove(std::shared_ptr<Graphic>& pGraphic);
    virtual std::shared_ptr<Graphic> GetChild(int iIndex);
    void Opereate();
};
#include "Graphic.h"
#include <iostream>


Graphic::Graphic(std::shared_ptr<Graphic> pParent) 
    : m_pParent(pParent)
{
}

 void Graphic::Add(std::shared_ptr<Graphic>& pGraphic)
 {
     std::cout<< "Leaf Cannot Add Childrren" << std::endl;
 }

 void Graphic::Remove(std::shared_ptr<Graphic>& pGraphic)
 {
     std::cout<< "Leaf Cannot Remove Childrren" << std::endl;
 }

 std::shared_ptr<Graphic> Graphic::GetChild(int iIndex)
 {
     std::cout<< "Leaf Cannot Get Child" << std::endl;
     return nullptr;
 }

 std::shared_ptr<Graphic>& Graphic::GetParent()
 {
     return m_pParent;
 }

 void Graphic::Opereate()
 {
     std::cout<< "Default Operate Executed" <<std::endl;
 }

 Circle::Circle(std::shared_ptr<Graphic> pParent)
     :Graphic(pParent)
 {
 }

 void Circle::Opereate()
 {
     std::cout << "Circle Operate Exeeuted" << std::endl;
 }

 Retangle::Retangle(std::shared_ptr<Graphic> pParent)
     :Graphic(pParent)
 {
 }

 void Retangle::Opereate()
 {
     std::cout << "Retangle Operate Exeeuted" << std::endl;
 }


  Line::Line(std::shared_ptr<Graphic> pParent)
     :Graphic(pParent)
 {
 }

 void Line::Opereate()
 {
     std::cout << "Line Operate Exeeuted" << std::endl;
 }


 Picture::Picture(std::shared_ptr<Graphic> pParent) 
    : Graphic(pParent)
{
}

 void Picture::Add(std::shared_ptr<Graphic>& pGraphic)
 {
     m_vecChildrenGraphics.push_back(pGraphic);

     std::cout<< "ChildrenGraphics Add Success" <<std::endl;
 }

 void Picture::Remove(std::shared_ptr<Graphic>& pGraphic)
 {
     for(auto iter=m_vecChildrenGraphics.begin();
         iter!=m_vecChildrenGraphics.end();++iter)
     {
         if((*iter) == pGraphic)
         {
             m_vecChildrenGraphics.erase(iter);
             std::cout<< "ChildrenGraphics remove Success"<<std::endl;
             break;
         }
     }
 }

 std::shared_ptr<Graphic> Picture::GetChild(int iIndex)
 {
     auto count = m_vecChildrenGraphics.size();
     if(iIndex <= count -1)
     {
         std::cout<< "Get Child Success" <<std::endl;
         return m_vecChildrenGraphics[iIndex];
     }

     return nullptr;
 }

 void Picture::Opereate()
 {
     for(auto iter=m_vecChildrenGraphics.begin();
         iter!=m_vecChildrenGraphics.end();++iter)
     {
         (*iter)->Opereate();
     }
 }
#include "Graphic.h"
#include <iostream>

int main()
{
    std::shared_ptr<Graphic> pPicture(new Picture(nullptr));

    std::shared_ptr<Graphic> pCircle(new Circle(pPicture));

    std::shared_ptr<Graphic> pRetangle(new Retangle(pPicture));

    std::shared_ptr<Graphic> pLine(new Line(pPicture));

    pPicture->Add(pCircle);
    pPicture->Add(pRetangle);
    pPicture->Add(pLine);

    pPicture->Opereate();

    if(nullptr == pPicture->GetParent())
    {
        std::cout<< "pPicture is root" <<std::endl;
    }

    auto pNode = pPicture->GetChild(0);

    auto pParent = pNode->GetParent();

    if(nullptr != pParent)
    {
        std::cout<<"Node has Parent"<<std::endl;
    }

    std::cout << "Parent Operate:"<<std::endl;

    pParent->Opereate();

    pNode->Opereate();

    pNode->Add(pNode);

    pNode->Remove(pNode);

    pNode->GetChild(0);


    pNode = pPicture->GetChild(1);

    if(nullptr != pNode->GetParent())
    {
        std::cout<<"Node has Parent"<<std::endl;
    }

    pNode->Opereate();

    pNode->Add(pNode);

    pNode->Remove(pNode);

    pNode->GetChild(0);

     pNode = pPicture->GetChild(2);

    if(nullptr != pNode->GetParent())
    {
        std::cout<<"Node has Parent"<<std::endl;
    }

    pNode->Opereate();

    pNode->Add(pNode);

    pNode->Remove(pNode);

    pNode->GetChild(0);

    pPicture->Remove(pCircle);
    pPicture->Remove(pRetangle);
    pPicture->Remove(pLine);


    while(1);

    



}

6.測試結果 

    

7.效果

  • 定義了包含基本對象和組合對象的類層次結構 基本對象可以被組合成更複雜的組合對象,而這個組合對象又可以被組合。
  • 簡化客戶代碼 客戶可以一致地使用組合結構和單個對象。
  • 使得容易增加新類型的組件 新定義的Composite或Leaf子類自動地與已有的結構和客戶代碼一起工作,客戶程式不需因新的Component類而改變。
  • 使設計變的更加一般化。

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • RAD Server是一個應用服務框架平臺,可快速構建和部署應用服務。RAD Server提供自動化的Delphi和C++ REST/ JSON API的 發佈與管理、企業資料庫集成中間件、智能物聯網設備軟體(IoT Edgeware)和一系列通用應用服務如用戶目錄和認證服務,消息推 送服務,室內/ ...
  • 建立Bank類,類中有變數double balance表示存款,Bank類的構造方法能初始化餘額,Bank類中有存款的方法cunKuan(double balance),取款的發方法withDrawal(double dAmount),當取款的數額大於存款時,拋出InsufficientFundsE ...
  • 第一步:獲取apache-tomcat-8.0.35-windows-x64.zip並解壓 第二步:從命令行進入bin目錄 第三步:執行命令:service.bat install,這樣就會將tomcat8.exe文件註冊成系統的服務。 第四步:啟動服務 第五步:在瀏覽器地址欄輸入http://lo ...
  • 按照C編譯器的約定調用函數時壓棧的順序是從右向左,並且返回值是保存在eax寄存器當中。這個命題本該是成立的,下麵用一個小程式來反彙編觀察執行過程: 代碼解釋一下,asm的代碼中movl %%eax, %0的意思是把寄存器eax的值賦值給咱們程式的eax變數當中。但為什麼執行結果卻是: z is 11 ...
  • 一、首先編寫微服務基礎項目framework 1、pom.xml 1 <?xml version="1.0" encoding="UTF-8"?> 2 <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w ...
  • 策略模式: 策略模式(Strategy Pattern):定義一系列演算法,將每一個演算法封裝起來,並讓它們可以相互替換。策略模式讓演算法獨立於使用它的客戶而變化,也稱為政策模式(Policy)。 策略模式是一種對象行為型模式。 模式動機: 完成一項任務,往往可以有多種不同的方式,每一種方式稱為一個策略, ...
  • 等寫完所有的代碼後,會在這裡給出整個項目的一個總覽圖。 技術介紹: 服務註冊和服務發現:consul 配置管理:consul 集群容錯:hystrix 計數監控:metrics 服務路由: 負載均衡: 服務通信:retrofit、okhttp 文檔輸出:swagger 日誌統計:logback+EL ...
  • 觀察者模式: 觀察者模式(Observer Pattern):定義對象間的一種一對多依賴關係,使得每當一個對象狀態發生改變時,其相關依賴對象皆得到通知並被自動更新。觀察者模式又叫做發佈-訂閱(Publish/Subscribe)模式、模型-視圖(Model/View)模式、源-監聽器(Source/ ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...