1、模式设计 c#创建型factory method名称 Factory Method结构意图定义一个用于创建对象的接口,让子类决定实例化哪一个类。Factory Method 使一个类的实例化延迟到其子类。适用性 当一个类不知道它所必须创建的对象的类的时候。 当一个类希望由它的子类来指定它所创建的对象的时候。 当类将创建对象的职责委托给多个帮助子类中的某一个,并且你希望将哪一个帮助子类是代理者这一信息局部化的时候。namespace FactoryMethod_DesignPatternusing System;/ These two classes could be part of a fra
2、mework,/ which we will call DP/ =class DPDocument abstract class DPApplication protected DPDocument doc;abstract public void CreateDocument();public void ConstructObjects()/ Create objects as needed/ . . ./ including documentCreateDocument(); abstract public void Dump();/ These two classes could be
3、part of an application / =class MyApplication : DPApplication override public void CreateDocument()doc = new MyDocument(); override public void Dump()Console.WriteLine(“MyApplication exists“); class MyDocument : DPDocument / / Summary description for Client./ public class Clientpublic static int Mai
4、n(string args)MyApplication myApplication = new MyApplication();myApplication.ConstructObjects();myApplication.Dump();return 0;http:/ 工厂方法(Factory Method)模式工厂方法(FactoryMethod)模式是类的创建模式,其用意是定义一个创建产品对象的工厂接口,将实际创建工作推迟到子类中。工厂方法模式是简单工厂模式的进一步抽象和推广。由于使用了多态性,工厂方法模式保持了简单工厂模式的优点,而且克服了它的缺点。在工厂方法模式中,核心的工厂类不再负责所
5、有产品的创建,而是将具体创建工作交给子类去做。这个核心类仅仅负责给出具体工厂必须实现的接口,而不接触哪一个产品类被实例化这种细节。这使得工厂方法模式可以允许系统在不修改工厂角色的情况下引进新产品。在 Factory Method 模式中,工厂类与产品类往往具有平行的等级结构,它们之间一一对应。二、 Factory Method 模式角色与结构:抽象工厂(Creator )角色:是工厂方法模式的核心,与应用程序无关。任何在模式中创建的对象的工厂类必须实现这个接口。具体工厂(Concrete Creator)角色:这是实现抽象工厂接口的具体工厂类,包含与应用程序密切相关的逻辑,并且受到应用程序调用
6、以创建产品对象。在上图中有两个这样的角色:BulbCreator 与 TubeCreator。抽象产品(Product )角色: 工厂方法模式所创建的对象的超类型,也就是产品对象的共同父类或共同拥有的接口。在上图中,这个角色是 Light。具体产品(Concrete Product)角色:这个角色实现了抽象产品角色所定义的接口。某具体产品有专门的具体工厂创建,它们之间往往一一对应。3、 程序举例:using System;public abstract class Lightpublic abstract void TurnOn();public abstract void TurnOff()
7、;public class BulbLight : Lightpublic override void TurnOn() Console.WriteLine(“Bulb Light is Turned on“); public override void TurnOff() Console.WriteLine(“Bulb Light is Turned off“); public class TubeLight : Lightpublic override void TurnOn() Console.WriteLine(“Tube Light is Turned on“); public ov
8、erride void TurnOff() Console.WriteLine(“Tube Light is Turned off“); public abstract class Creatorpublic abstract Light factory();public class BulbCreator : Creatorpublic override Light factory() return new BulbLight(); public class TubeCreator : Creatorpublic override Light factory() return new Tub
9、eLight(); public class Clientpublic static void Main()Creator c1 = new BulbCreator();Creator c2 = new TubeCreator();Light l1 = c1.factory();Light l2 = c2.factory();l1.TurnOn();l1.TurnOff();Console.WriteLine(“-“);l2.TurnOn();l2.TurnOff();工厂方法的活动序列图活动过程包括:客户端创建 BulbCreator 对象,客户端持有此对象的类型是 Creator,而实际类
10、型是BulbCreator。然后客户端调用 BulbCreator 的 factory 方法,之后 BulbCreator 调用BulbLight 的构造函数创造出产品 BulbLight 对象。四、 工厂方法模式与简单工厂模式工厂方法模式与简单工厂模式再结构上的不同不是很明显。工厂方法类的核心是一个抽象工厂类,而简单工厂模式把核心放在一个具体类上。工厂方法模式之所以有一个别名叫多态性工厂模式是因为具体工厂类都有共同的接口,或者有共同的抽象父类。当系统扩展需要添加新的产品对象时,仅仅需要添加一个具体对象以及一个具体工厂对象,原有工厂对象不需要进行任何修改,也不需要修改客户端,很好的符合了“开放
11、封闭“原则。而简单工厂模式在添加新产品对象后不得不修改工厂方法,扩展性不好。工厂方法模式退化后可以演变成简单工厂模式。五、 Factory Method 模式演化使用接口或抽象类抽象工厂角色和抽象场频角色都可以选择由接口或抽象类实现。使用多个工厂方法抽象工厂角色可以规定出多于一个的工厂方法,从而使具体工厂角色实现这些不同的工厂方法,这些方法可以提供不同的商业逻辑,以满足提供不同的产品对象的任务。产品的循环使用工厂方法总是调用产品类的构造函数以创建一个新的产品实例,然后将这个实例提供给客户端。而在实际情形中,工厂方法所做的事情可以相当复杂。一个常见的复杂逻辑就是循环使用产品对象。工厂对象将已经创
12、建过的产品登记到一个聚集中,然后根据客户所请求的产品状态,向聚集查询。如果有满足要求的产品对象,就直接将产品返回客户端;如果聚集中没有这样的产品对象,那么就创建一个新的满足要求的产品对象,然后将这个对象登记到聚集中,再返还给客户端。“享元模式(Flyweight Pattern)“ 就是这样一个模式。多态性的丧失和模式的退化一个工厂方法模式的实现依赖于工厂角色和产品角色的多态性。在有些情况下,这个模式可以出现退化。工厂方法返回的类型应当是抽象类型,而不是具体类型。调用工厂方法的客户端应当依赖抽象产品编程,而不是具体产品。如果工厂仅仅返回一个具体产品对象,便违背了工厂方法的用意,发生退化,这时就
13、不再是工厂模式了。工厂的等级结构:工厂对象应当有一个抽象的超类型。如果等级结构中只有一个具体工厂类的话,抽象工厂就可以省略,发生了退化。六、 Factory Method 模式与其它模式的关系与工厂方法模式有关的模式还包括:模板方法模式、MVC 模式、享元模式、备忘录模式7、 另外一个例子/ Factory Method pattern - Real World example using System;using System.Collections;/ “Product“abstract class Page/ “ConcreteProduct“class SkillsPage : Pag
14、e/ “ConcreteProduct“class EducationPage : Page/ “ConcreteProduct“class ExperiencePage : Page/ “ConcreteProduct“class IntroductionPage : Page/ “ConcreteProduct“class ResultsPage : Page/ “ConcreteProduct“class ConclusionPage : Page/ “ConcreteProduct“class SummaryPage : Page/ “ConcreteProduct“class Bib
15、liographyPage : Page/ “Creator“abstract class Document/ Fieldsprotected ArrayList pages = new ArrayList();/ Constructorpublic Document()this.CreatePages();/ Propertiespublic ArrayList Pagesget return pages; / Factory Methodabstract public void CreatePages();/ “ConcreteCreator“class Resume : Document
16、/ Factory Method implementationoverride public void CreatePages()pages.Add( new SkillsPage() );pages.Add( new EducationPage() );pages.Add( new ExperiencePage() );/ “ConcreteCreator“class Report : Document/ Factory Method implementationoverride public void CreatePages()pages.Add( new IntroductionPage
17、() );pages.Add( new ResultsPage() );pages.Add( new ConclusionPage() );pages.Add( new SummaryPage() );pages.Add( new BibliographyPage() );/ / FactoryMethodApp test/ class FactoryMethodApppublic static void Main( string args )Document docs = new Document 2 ;/ Note: constructors call Factory Methoddocs0 = new Resume();docs1 = new Report();/ Display document pagesforeach( Document document in docs )Console.WriteLine( “ + document + “ - “ );foreach( Page page in document.Pages )Console.WriteLine( “ “ + page );