概述 工廠方法模式(FactoryMethod),定義一個創建產品對象的工廠介面,讓工廠子類決定實例化那一個產品類。我們把被創建的對象稱為“產品”,把創建產品的對象稱為“工廠”。如果要創建的產品不多,只要一個工廠類就可以完成,這種模式叫“簡單工廠模式”,它不屬於 23 種經典設計模式,它的缺點是增加 ...
概述
工廠方法模式(FactoryMethod),定義一個創建產品對象的工廠介面,讓工廠子類決定實例化那一個產品類。我們把被創建的對象稱為“產品”,把創建產品的對象稱為“工廠”。如果要創建的產品不多,只要一個工廠類就可以完成,這種模式叫“簡單工廠模式”,它不屬於 23 種經典設計模式,它的缺點是增加新產品時會違背“開閉原則”(可以通過反射剋服該缺點)。
工廠方法模式是對簡單工廠模式的進一步抽象化,其好處是可以使系統在不修改原來代碼的情況下引進新的產品,即滿足開閉原則。
需求案例
我們先來看一看這樣一個案例,小學生玩的四則運算,當客戶輸入兩個數,然後輸入一個操作符,獲得結果!我們先用簡單工廠寫下邏輯代碼,驗證下簡單工廠的缺點。
簡單工廠代碼
/// <summary> /// 靜態工作方法 實際上就是把創建對象的過程放到靜態方法裡面 /// </summary> public class CalFactory { public static ICalculator GetCalculator(string oper) { ICalculator? calculator = null; switch (oper) { case "*": calculator = new Mul(); break; case "+": calculator = new Add(); break; case "-": calculator = new Sub(); break; case "/": calculator = new Div(); break; } return calculator; } }
public interface ICalculator { double GetResult(double d1,double d2); } public class Add : ICalculator { public double GetResult(double d1, double d2) { return d1+ d2; } } public class Sub : ICalculator { public double GetResult(double d1, double d2) { return d1-d2; } } public class Mul : ICalculator { public double GetResult(double d1, double d2) { return d1* d2; } } public class Div : ICalculator { public double GetResult(double d1, double d2) { return d1 / d2; }
//C# 控制台調用 Console.WriteLine("簡單工廠設計模式!"); Console.WriteLine("請輸入操作數1"); var d1 = Convert.ToDouble(Console.ReadLine()); Console.WriteLine("請輸入操作數2"); var d2 = Convert.ToDouble(Console.ReadLine()); ICalculator calculator = null; Console.WriteLine("請輸入操作符"); string flag=Console.ReadLine(); calculator= CalFactory.GetCalculator(flag); double reslut= calculator.GetResult(d1,d2); Console.WriteLine(reslut);
這個需求的代碼邏輯很簡單,一看基本就會,我們主要說下簡單工廠模式的優缺點;
優點:
1、簡單工廠設計模式解決了客戶端直接依賴具體對象的問題,客戶端消除了創建對象的責任,僅僅承擔使用的責任。簡單工廠模式實現了對責任的分割;
2、簡單工廠起到了代碼復用的作用;
缺點:
1、系統擴展困難,一旦加入新功能,就必須要修改工廠邏輯。破壞了開閉原則;
2、簡單工廠集合了多有創建對象的邏輯,一旦不能正常工作,會導致整個系統出問題;
工廠方法代碼
還是上面的需求,我們定義一個創建對象的介面,讓子類決定實例化哪一個類。
public interface ICalFactory { ICalculator GetCalculator(); } public class MulFactory : ICalFactory { public ICalculator GetCalculator() { return new Mul(); } } public class AddFactory : ICalFactory { public ICalculator GetCalculator() { return new Add(); } } public class SubFactory : ICalFactory { public ICalculator GetCalculator() { return new Sub(); } } public class DivFactory : ICalFactory { public ICalculator GetCalculator() { return new Div(); } }
public interface ICalculator { double GetResult(double d1, double d2); } public class Add : ICalculator { public double GetResult(double d1, double d2) { return d1 + d2; } } public class Sub : ICalculator { public double GetResult(double d1, double d2) { return d1 - d2; } } public class Mul : ICalculator { public double GetResult(double d1, double d2) { return d1 * d2; } } public class Div : ICalculator { public double GetResult(double d1, double d2) { return d1 / d2; } }
//C# 控制台調用 Console.WriteLine("請輸入操作數1"); var d1 = Convert.ToDouble(Console.ReadLine()); Console.WriteLine("請輸入操作數2"); var d2 = Convert.ToDouble(Console.ReadLine()); Console.WriteLine("請輸入操作符"); string flag = Console.ReadLine(); ICalFactory? calFactory = null; switch (flag) { case "*": calFactory = new MulFactory(); break; case "+": calFactory = new AddFactory(); break; case "-": calFactory = new SubFactory(); break; case "/": calFactory = new DivFactory(); break; } ICalculator calculator=calFactory?.GetCalculator(); var result= calculator.GetResult(d1, d2); Console.WriteLine(result);
我們目前是使用了工廠方法設計模式來實現了需求,但能看到源碼裡面有一坨switch case,我們接下來通過反射處理switch case
沒有switch case的工廠方法代碼
public class OperFactory:Attribute { public string Oper { get; } public OperFactory(string value) { Oper=value; } }
public class ReflectionFactory { //根據用戶輸入的操作符 返回一個對象 Dictionary<string, ICalFactory> dic=new Dictionary<string, ICalFactory>(); public ReflectionFactory() { //拿到當前正在運行的程式集 Assembly assembly= Assembly.GetExecutingAssembly(); foreach (var item in assembly.GetTypes()) { //IsAssignableFrom 表示item繼承了ICalFactory或實現了ICalFactory if (typeof(ICalFactory).IsAssignableFrom(item)&&!item.IsInterface) { OperFactory of= item.GetCustomAttribute<OperFactory>(); if (of != null) { //給鍵值對集合賦值 dic[of.Oper] = Activator.CreateInstance(item) as ICalFactory; } } } } public ICalFactory GetFac(string s) { if (dic.ContainsKey(s)) { return dic[s]; } return null; } }
public interface ICalFactory { ICalculator GetCalculator(); } [OperFactory("*")] public class MulFactory : ICalFactory { public ICalculator GetCalculator() { return new Mul(); } } [OperFactory("+")] public class AddFactory : ICalFactory { public ICalculator GetCalculator() { return new Add(); } } [OperFactory("-")] public class SubFactory : ICalFactory { public ICalculator GetCalculator() { return new Sub(); } } [OperFactory("/")] public class DivFactory : ICalFactory { public ICalculator GetCalculator() { return new Div(); } }
//C# 控制台調用 ReflectionFactory reflectionFactory = new ReflectionFactory(); ICalFactory calFactory2= reflectionFactory.GetFac(flag); var result2= calFactory2.GetCalculator().GetResult(d1, d2); Console.WriteLine(result2);
總結
一個四則運算的需求,分別採用簡單工廠和工廠方法實現,其實各有優缺點,使用哪種設計模式主要取決於自己的業務,有何疑問,歡迎交流。
作者:realyrare
出處:https://www.cnblogs.com/mhg215/
聲援博主:如果您覺得文章對您有幫助,請點擊文章末尾的【關註我】吧!
別忘記點擊文章右下角的【推薦】支持一波。~~~///(^v^)\\\~~~ .
本文版權歸作者和博客園共有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責任的權利。
如果您有其他問題,也歡迎關註我下方的公眾號,可以聯繫我一起交流切磋!