模式:工程化實現及擴展——工廠模式

来源:https://www.cnblogs.com/zhiyong-ITNote/archive/2019/07/11/11173525.html
-Advertisement-
Play Games

相比較傳統的工廠模式IFactory/Concrete Factory會反覆引用並編譯代碼 但是作為開發人員,我們更希望的是少修改代碼,儘量從配置著手也就是設計模式的根本原則之一:開放封閉原則。如果我要增加新的產品,那麼修改就比較大了,對於業務來講還是可以接受的。但是如果可以做到不修改代碼是最好的。 ...


相比較傳統的工廠模式IFactory/Concrete Factory會反覆引用並編譯代碼
但是作為開發人員,我們更希望的是少修改代碼,儘量從配置著手也就是設計模式的根本原則之一:開放封閉原則。如果我要增加新的產品,那麼修改就比較大了,對於業務來講還是可以接受的。但是如果可以做到不修改代碼是最好的。上一份工作中,我印象最深的一句話就是我上司對我說的"能不改代碼就別改,能寫進配置里的就寫到配置里"。因此我們將要增加的工廠類寫到配置裡面。如此,新的產品類型和工廠類型即便在系統上線後仍可以通過修改配置文件的方式不斷補充。但是,還有一個問題,我們仍然需要為每"類"抽象產品定製特定的工廠介面並實現之,也就是"多頭管理"問題。泛型可以用來解決這個問題,我們定義一個泛型工廠即可。代碼如下:

/// <summary>
/// 工廠介面定義
/// </summary>
/// <remarks>
/// TTarget: 抽象產品類型
/// TSource: 具體產品類型
/// </remarks>
public interface IFactory
{
    #region config and register type mapping

    /// <summary>
    /// 如果需要同時載入配置文件中定義的映射關係,可以按照SRP的原則定義獨立的配置類型。
    /// 由該配置類型調用這兩個介面為Factory載入配置信息
    /// </summary>

    IFactory RegisterType<TTarget, TSource>();  // 註入產品
    IFactory RegisterType<TTarget, TSource>(string name);   // 註入產品

    #endregion

    #region factory method

    TTarget Create<TTarget>();
    TTarget Create<TTarget>(string name);

    #endregion
}

public sealed class TypeRegistry
{
    /// <summary>
    /// default name in type mappings
    /// </summary>
    readonly string DefaultName = Guid.NewGuid().ToString();

    /// <summary>
    /// Type        :   TTarget, 抽象產品類型
    /// IDictionary<string ,Type>
    ///     string  :   name
    ///     Type    :   TSource, 具體產品類型
    /// </summary>
    IDictionary<Type, IDictionary<string, Type>> registry =
        new Dictionary<Type, IDictionary<string, Type>>();

    public void RegisterType(Type targetType, Type sourceType)
    {
        RegisterType(targetType, sourceType, DefaultName);
    }

    public void RegisterType(Type targetType, Type sourceType, string name)
    {
        if(targetType == null) throw new ArgumentNullException("targetType");
        if(sourceType == null) throw new ArgumentNullException("sourceType");
        if(string.IsNullOrEmpty(name)) throw new ArgumentNullException("name");

        if (!registry.TryGetValue(targetType, out IDictionary<string, Type> subDictionary))
        {
            subDictionary = new Dictionary<string, Type>
            {
                { name, sourceType }
            };
            registry.Add(targetType, subDictionary);
        }
        else
        {
            if (subDictionary.ContainsKey(name))
                throw new Exception($"{name}重覆");
            subDictionary.Add(name, sourceType);
        }
    }

    public Type this[Type targetType, string name]
    {
        get
        {
            if (targetType == null) throw new ArgumentNullException("targetType");
            if (string.IsNullOrEmpty(name)) throw new ArgumentNullException("name");
            if (registry.Count() == 0)
                return null;

            return 
                (registry.Where(x => x.Key == targetType)).FirstOrDefault().Value
                    .Where(x => string.Equals(name, x.Key)).FirstOrDefault().Value;
        }
    }

    public Type this[Type targetType]
    {
        get { return this[targetType, DefaultName]; }
    }
}

public class Factory : IFactory
{
    protected TypeRegistry registry = new TypeRegistry();

    #region IFactory Members

    public IFactory RegisterType<TTarget, TSource>()
    {
        registry.RegisterType(typeof(TTarget), typeof(TSource));
        return this;
    }

    public IFactory RegisterType<TTarget, TSource>(string name)
    {
        registry.RegisterType(typeof(TTarget), typeof(TSource), name);
        return this;
    }

    public TTarget Create<TTarget>()
    {
        return (TTarget)Activator.CreateInstance(registry[typeof(TTarget)]);
    }

    public TTarget Create<TTarget>(string name)
    {
        return (TTarget)Activator.CreateInstance(registry[typeof(TTarget), name]);
    }

    #endregion
}

上面的示例表明新的工廠類型不僅可以完成經典工廠方法模式所希望實現的各項要求,也滿足抽象工廠的要求,同時他可以作為整個項目一個獨立的而且是唯一的工廠入口,供項目中各子系統訪問和使用。原因在於它的底層將工廠介面與抽象產品類型的依賴關係變成基於CLR"萬能工廠"類型Activator基於參數Type的構造。
工廠管理的是其內的產品。我們的工廠介面IFactory有兩個功能,一個是往工廠中註入產品,一個是創建指定產品的實例。藉助RegisterType將配置文件中定義的類型映射方希載入到新的具體工廠類型中,也就是重載函數中的參數(name)。我們通過字典Dictionary來管理維護工廠內的產品,將抽象產品也就是介面或是抽象類作為key,要考慮到同一介面可以有多個不同的實現,因此我們再維護一個實現類的字典,使用一個唯一的標識作為key就行,value就是實現類。


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

-Advertisement-
Play Games
更多相關文章
  • 本文介紹TypeScript中泛型的概念,以簡單直白的方式,向那些不瞭解此概念,但渴望在工作中使用它的開發者,提供入門指導。 ...
  • 以上是效果圖,本圖表使用d3.js v4製作。圖表主要功能是在六邊形格子中顯示數據,點擊底部圖標可以切換指定格子高亮顯示,圖表可以隨瀏覽器任意縮放。 1.圖表的主體結構是由正六邊形組成,使用d3生成六邊形可以使用d3-hexbin.js,生成六邊形比較方便,只要給定中心點坐標和半徑即可生成六邊形路徑 ...
  • 問題一、微信瀏覽器中無法使用reload重載文檔【VUE框架】 問題分析: 微信不支持location.reload()方法,在微信瀏覽器中會失效 Vue中的路由跳轉是類似於ajax局部刷新,因此使用location.href=‘xxx+時間戳’ 這種方法時,頁面不會重載 Vue自帶的this.$r ...
  • 1.地圖繪製過程原理 給定範圍邊界經緯度數據,再給它個名字就構成了繪製地圖的基礎。也就是說,你可以繪製任意形狀的地圖版塊。 2.地圖數據生成 中國以及省市縣等地圖的基礎數據可以從這裡生成與下載。 "http://datav.aliyun.com/tools/atlas" 有了地圖範圍數據,在 ech ...
  • 哈哈,久違了各位。我又回來了,最近在做畢設,所以難免會遇到很多問題,需要解決很多問題,在萬能的博友幫助下,終於解決了Element-ui中輪播圖的圖片高度問題,話不多說上代碼。 那個axios的使用不重要,大致思路就是頁面渲染前拿到當前視窗的寬度*圖片比例給輪播圖一個初始的高度,當窗後大小發生變化時 ...
  • 本人剛學先上鏈接(別人寫的挺好的)後期同步補上😄!!! 網站鏈接:https://blog.csdn.net/baozhiqiangjava/article/details/81178694 GitHub:https://github.com/JsAaron/jQuery ...
  • SpringCloud系列教程 | 第十二篇:Spring Cloud Gateway初探 Springboot: 2.1.6.RELEASE SpringCloud: Greenwich.SR1 如無特殊說明,本系列教程全採用以上版本 前面我們在聊服務網關Zuul的時候提到了Gateway,那麼Z ...
  • "JavaScript 設計模式基礎(一)" "小菜鳥的個人博客" 原型模式 在以類為中心的面向對象編程語言中,類和對象的關係就像鑄模和鑄件的關係,對象總是從類中創建。而原型編程中,類不是必須的,對象未必從類中創建而來,可以拷貝另一個對象而變成新對象 從設計模式角度講,原型模式是用於創建對象的一種模 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...