C++ 深入淺出工廠模式(進階篇)

来源:https://www.cnblogs.com/xiaolincoding/archive/2019/09/15/11524401.html
-Advertisement-
Play Games

介紹 前文初始篇 "C++ 深入淺出工廠模式(初始篇)" ,主要闡述了簡單工廠模式、工廠方法模式和抽象工廠模式的結構、特點和缺陷等。以上三種方式,在新增產品時,要麼修改工廠類,要麼需新增具體的工廠類,說明工廠類的封裝性還不夠好。 本文進階篇,主要是將工廠類的封裝性提高,達到新增產品時,也不需要修改工 ...


介紹

前文初始篇C++ 深入淺出工廠模式(初始篇),主要闡述了簡單工廠模式、工廠方法模式和抽象工廠模式的結構、特點和缺陷等。以上三種方式,在新增產品時,要麼修改工廠類,要麼需新增具體的工廠類,說明工廠類的封裝性還不夠好。

本文進階篇,主要是將工廠類的封裝性提高,達到新增產品時,也不需要修改工廠類,不需要新增具體的工廠類。封裝性高的工廠類特點是擴展性高、復用性也高。

模板工廠

針對工廠方法模式封裝成模板工廠類,那麼這樣在新增產品時,是不需要新增具體的工廠類,減少了代碼的編寫量。

UML圖:

模板工廠代碼:
  • ShoesClothe,分別為鞋子和衣服的抽象類(基類)
  • NiKeShoesUniqloClothe,分別為耐克鞋子和優衣庫衣服具體產品類。
// 基類 鞋子
class Shoes
{
public:
    virtual void Show() = 0;
    virtual ~Shoes() {}
};

// 耐克鞋子
class NiKeShoes : public Shoes
{
public:
    void Show()
    {
        std::cout << "我是耐克球鞋,我的廣告語:Just do it" << std::endl;
    }
};

// 基類 衣服
class Clothe
{
public:
    virtual void Show() = 0;
    virtual ~Clothe() {}
};

// 優衣庫衣服
class UniqloClothe : public Clothe
{
public:
    void Show()
    {
        std::cout << "我是優衣庫衣服,我的廣告語:I am Uniqlo" << std::endl;
    }
};
  • AbstractFactory為抽象模板工廠類,其中模板參數:AbstractProduct_t 產品抽象類,如ShoesClothe
  • ConcreteFactory為具體模板工廠類,其中模板參數:AbstractProduct_t 產品抽象類(如ShoesClothe),ConcreteProduct_t 產品具體類(如NiKeShoesUniqloClothe
// 抽象模板工廠類
// 模板參數:AbstractProduct_t 產品抽象類
template <class AbstractProduct_t>
class AbstractFactory
{
public:
    virtual AbstractProduct_t *CreateProduct() = 0;
    virtual ~AbstractFactory() {}
};

// 具體模板工廠類
// 模板參數:AbstractProduct_t 產品抽象類,ConcreteProduct_t 產品具體類
template <class AbstractProduct_t, class ConcreteProduct_t>
class ConcreteFactory : public AbstractFactory<AbstractProduct_t>
{
public:
    AbstractProduct_t *CreateProduct()
    {
        return new ConcreteProduct_t();
    }
};
  • main函數,根據不同類型的產品,構造對應的產品的工廠對象,便可通過對應產品的工廠對象創建具體的產品對象。
int main()
{
    // 構造耐克鞋的工廠對象
    ConcreteFactory<Shoes, NiKeShoes> nikeFactory;
    // 創建耐克鞋對象
    Shoes *pNiKeShoes = nikeFactory.CreateProduct();
    // 列印耐克鞋廣告語
    pNiKeShoes->Show();

    // 構造優衣庫衣服的工廠對象
    ConcreteFactory<Clothe, UniqloClothe> uniqloFactory;
    // 創建優衣庫衣服對象
    Clothe *pUniqloClothe = uniqloFactory.CreateProduct();
    // 列印優衣庫廣告語
    pUniqloClothe->Show();

    // 釋放資源
    delete pNiKeShoes;
    pNiKeShoes = NULL;

    delete pUniqloClothe;
    pUniqloClothe = NULL;

    return 0;
}
  • 輸出結果:
[root@lincoding factory]# ./templateFactory 
我是耐克球鞋,我的廣告語:Just do it
我是優衣庫衣服,我的廣告語:I am Uniqlo

產品註冊模板類+單例工廠模板類

前面的模板工廠雖然在新增產品的時候,不需要新增具體的工廠類,但是缺少一個可以統一隨時隨地獲取指定的產品對象的類。

還有改進的空間,我們可以把產品註冊的對象用std::map的方式保存,通過key-valve的方式可以輕鬆簡單的獲取對應的產品對象實例。

實現大致思路:

  • 把產品註冊的功能封裝成產品註冊模板類。註冊的產品對象保存在工廠模板類的std::map,便於產品對象的獲取。

  • 把獲取產品對象的功能封裝成工廠模板類。為了能隨時隨地獲取指定產品對象,則把工廠設計成單例模式。

UML圖:

產品註冊模板類+單例工廠模板類:
  • IProductRegistrar為產品註冊抽象類,模板參數 ProductType_t 表示的類是產品抽象類(如ShoesClothe)。提供了產品對象創建的純虛函數CreateProduct
  • ProductFactory為工廠模板類,模板參數 ProductType_t 表示的類是產品抽象類(如ShoesClothe)。用於保存註冊產品對象到std::map中和獲取對應的產品對象。
  • ProductRegistrar為產品註冊模板類,模板參數 ProductType_t 表示的類是產品抽象類(如ShoesClothe),ProductImpl_t 表示的類是具體產品(如NikeShoesUniqloClothe)。用於註冊產品到工廠類和創建產品實例對象。
// 基類,產品註冊模板介面類
// 模板參數 ProductType_t 表示的類是產品抽象類
template <class ProductType_t>
class IProductRegistrar
{
public:
   // 獲取產品對象抽象介面
   virtual ProductType_t *CreateProduct() = 0;

protected:
   // 禁止外部構造和虛構, 子類的"內部"的其他函數可以調用
   IProductRegistrar() {}
   virtual ~IProductRegistrar() {}

private:
   // 禁止外部拷貝和賦值操作
   IProductRegistrar(const IProductRegistrar &);
   const IProductRegistrar &operator=(const IProductRegistrar &);
};

// 工廠模板類,用於獲取和註冊產品對象
// 模板參數 ProductType_t 表示的類是產品抽象類
template <class ProductType_t>
class ProductFactory
{
public:
   // 獲取工廠單例,工廠的實例是唯一的
   static ProductFactory<ProductType_t> &Instance()
   {
      static ProductFactory<ProductType_t> instance;
      return instance;
   }

   // 產品註冊
   void RegisterProduct(IProductRegistrar<ProductType_t> *registrar, std::string name)
   {
      m_ProductRegistry[name] = registrar;
   }

   // 根據名字name,獲取對應具體的產品對象
   ProductType_t *GetProduct(std::string name)
   {
      // 從map找到已經註冊過的產品,並返回產品對象
      if (m_ProductRegistry.find(name) != m_ProductRegistry.end())
      {
         return m_ProductRegistry[name]->CreateProduct();
      }

      // 未註冊的產品,則報錯未找到
      std::cout << "No product found for " << name << std::endl;

      return NULL;
   }

private:
   // 禁止外部構造和虛構
   ProductFactory() {}
   ~ProductFactory() {}

   // 禁止外部拷貝和賦值操作
   ProductFactory(const ProductFactory &);
   const ProductFactory &operator=(const ProductFactory &);

   // 保存註冊過的產品,key:產品名字 , value:產品類型
   std::map<std::string, IProductRegistrar<ProductType_t> *> m_ProductRegistry;
};

// 產品註冊模板類,用於創建具體產品和從工廠里註冊產品
// 模板參數 ProductType_t 表示的類是產品抽象類(基類),ProductImpl_t 表示的類是具體產品(產品種類的子類)
template <class ProductType_t, class ProductImpl_t>
class ProductRegistrar : public IProductRegistrar<ProductType_t>
{
public:
   // 構造函數,用於註冊產品到工廠,只能顯示調用
   explicit ProductRegistrar(std::string name)
   {
      // 通過工廠單例把產品註冊到工廠
      ProductFactory<ProductType_t>::Instance().RegisterProduct(this, name);
   }

   // 創建具體產品對象指針
   ProductType_t *CreateProduct()
   {
      return new ProductImpl_t();
   }
};
  • main函數,通過ProductRegistrar註冊各種不同類型產品,在統一由ProductFactory單例工廠獲取指定的產品對象。
int main()
{
   // ========================== 生產耐克球鞋過程 ===========================//
   // 註冊產品種類為Shoes(基類),產品為NiKe(子類)到工廠,產品名為nike
   ProductRegistrar<Shoes, NiKeShoes> nikeShoes("nike");
   // 從工廠獲取產品種類為Shoes,名稱為nike的產品對象
   Shoes *pNiKeShoes = ProductFactory<Shoes>::Instance().GetProduct("nike");
   // 顯示產品的廣告語
   pNiKeShoes->Show();
   // 釋放資源
   if (pNiKeShoes)
   {
      delete pNiKeShoes;
   }

   // ========================== 生產優衣庫衣服過程 ===========================//
   // 註冊產品種類為Clothe(基類),產品為UniqloClothe(子類)到工廠,產品名為uniqlo
   ProductRegistrar<Clothe, UniqloClothe> adidasShoes("uniqlo");
   // 從工廠獲取產品種類為Shoes,名稱為adidas的產品對象
   Clothe *pUniqloClothe = ProductFactory<Clothe>::Instance().GetProduct("uniqlo");
   // 顯示產品的廣告語
   pUniqloClothe->Show();
   // 釋放資源
   if (pUniqloClothe)
   {
      delete pUniqloClothe;
   }

   return 0;
}
  • 輸出結果:
[root@lincoding factory]# ./singleFactory 
我是耐克球鞋,我的廣告語:Just do it
我是優衣庫衣服,我的廣告語:I am Uniqlo

總結

將工廠方法模式改良成模板工廠,雖然可以解決產品新增時,不需要新增具體工廠類,但是缺少一個可以隨時隨地獲取產品對象的方式,說明還有改進的空間。

將模板工廠改良成產品註冊模板類+單例工廠模板類,產品註冊模板類用於註冊不同類型的產品,單例工廠模板類用於獲取指定已註冊的產品對象。這種方式,可以把工廠模式中產品的註冊和獲取的主要功能很好的抽象成兩個類,並且使用單例模式使得工廠類可以隨時隨地獲取已註冊的產品對象。

所以產品註冊模板類+單例工廠模板類的工廠模式,達到了開閉法則,並且擴展性高和封裝度高。

PS:想學習更多單例模式,可以參考C++ 線程安全的單例模式總結文章閱讀。


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

-Advertisement-
Play Games
更多相關文章
  • 一、什麼是promise?我們用promise解決什麼樣的問題 promise是非同步編程的一種解決方案:從語法上來說,Promise是一個對象,從他可以獲取非同步操作的信息;從本意上講,它是承諾,它承諾過一段時間會給你一個答覆。Promise有三種狀態:pending(等待態)、fulfiled(成功 ...
  • iconfont的使用: "https://www.cnblogs.com/changxin7/p/11479216.html" Bootstrap介紹 Bootstrap是Twitter開源的基於HTML、CSS、JavaScript的前端框架。 它是為實現快速開發Web應用程式而設計的一套前端工 ...
  • 分組元素用於對頁面中的內容進行分組。 figure元素和figcaption元素 figure 元素用於定義獨立的流內容(圖像、圖表、照片、代碼等),一般指一個獨立的單元。 figure 元素的內容應該與主內容相關,但如果被刪除,也 不會對文檔流產生影響 。 figcaption 元素用於為figu ...
  • 1.let作用域局限於當前代碼塊 代碼1: let作用域僅限於當前代碼塊,而var的作用域是全局的 2.let作用域不會被提升 代碼2: let作用域不會被提升,而var作用域會被提升 代碼2相當於: 3.let不能被重覆定義 代碼3: 上面這段代碼運行會報錯:Identifier 'str2' h ...
  • 前言 vue通信手段有很多種,props/emit、vuex、event bus、provide/inject 等。還有一種通信方式,那就是 $attrs 和 $listeners,之前早就聽說這兩個api,趁著有空來補補。這種方式挺優雅,使用起來也不賴。下麵例子都會通過父、子、孫子,三者的關係來說 ...
  • [TOC] 1.DOM樹介紹 DOM:文檔對象模型。DOM 為文檔提供了結構化表示,並定義瞭如何通過腳本來訪問文檔結構。目的其實就是為了能讓js操作html元素而制定的一個規範。 DOM就是由節點組成的:HTML載入完畢,渲染引擎會在記憶體中把HTML文檔,生成一個DOM樹。 在HTML當中,一切都是 ...
  • 一、使用方法: 因為map標簽是與img標簽綁定使用的,所以我們需要給map標簽添加ID和name屬性,讓img標簽中的usemap屬性引用map標簽中的id或者name屬性(由於瀏覽器的不同,usemap屬性接收二者之一的值,所以通常name和id屬性二者都寫,值相同),並配合area標簽進行使用 ...
  • [TOC] 1.javascript介紹 1.1Web前端有三層: HTML:從語義的角度,描述頁面 結構 CSS:從審美的角度,描述 樣式 (美化頁面) JavaScript:從交互的角度,描述 行為 (提升用戶體驗) 1.2其中JavaScript基礎又分為三個部分: ECMAScript:Ja ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...