在五大设计原则的基础上经过GOF(四人组)的总结,得出了23种经典设计模式,其中分为三大类:创建型(5种)、结构型(7种)、行为型(11种)。今天对创建型中的工厂方法(FactoryMethod)模式的思想进行了一下复习和实践,在此也做一下记录。同样,理解或实践不到位的地方,希望走过路过的看官指正一下!

  同样先来看看工厂方法(FactoryMethod)模式的定义:

  Define an interface for creating an object, but let subclasses decide which class to instantiate.

  意思就是说:定义一个用于创建对象的接口,但让子类决定要实例化哪个类。

  也就是工厂方法(FactoryMethod)模式允许将产品类的实例化推迟到具体的创建者子类,由创建者子类决定实例化哪一个产品类。我们同样以汽车的生产作为讲解该模式的例子,因为汽车生产从宏观上来说也是特别符合这个模式的。我要一辆汽车,什么品牌的我没有要求,符合是汽车这个标准就行了。那么世面上不同的汽车生产商就会用自己的生产流程生产出符合汽车这个标准的不同品牌汽车。同样,我们也撸起袖子码一码!因为该模式没有强调待生产的产品类是复杂的,同样也为了减少大家的代码阅读,这次我们把汽车相关类定义的更简单一点!

  由于是让子类决定要实例化哪个产品类,那么高层的类或客户代码肯定是不需要知道有哪些具体产品类的,也就是得为众多具体的产品类制定一个产品标准(可以是接口,也可以是抽象类),这里我们定义一个汽车接口——ICar,它只有一个启动接口方法:

    /// <summary>
/// 汽车接口
/// </summary>
public interface ICar
{
/// <summary>
/// 启动汽车
/// </summary>
/// <returns>是否启动成功</returns>
bool Start();
}

  

  有了汽车产品标准(ICar),我们在此基础上定义两个实现了该标准的具有汽车产品类——奔驰(Benz)和宝马(BWM),它们对汽车产品标准中的启动标准做了不一样的实现(机械钥匙启动和按键启动):

    /// <summary>
/// 奔驰汽车
/// </summary>
[Serializable]
public class BenzCar : ICar
{
private readonly string name = "Benz";
private readonly string model;
/// <summary>
/// 车名
/// </summary>
public string Name
{
get { return name; }
}
/// <summary>
/// 型号
/// </summary>
public string Model
{
get { return model; }
}
/// <summary>
/// 初始化奔驰汽车
/// </summary>
public BenzCar() { }
/// <summary>
/// 初始化奔驰汽车(带简单参数)
/// </summary>
/// <param name="model"></param>
public BenzCar(string model)
{
this.model = model;
}
/// <summary>
/// 启动奔驰汽车
/// </summary>
/// <returns>是否启动成功</returns>
public bool Start()
{
Console.WriteLine("【{0} {1}】通过机械钥匙启动!", Name, Model);
return true;
}
} /// <summary>
/// 宝马汽车
/// </summary>
[Serializable]
public class BwmCar : ICar
{
private readonly string name = "BWM";
private readonly string model;
/// <summary>
/// 车名
/// </summary>
public string Name
{
get { return name; }
}
/// <summary>
/// 型号
/// </summary>
public string Model
{
get { return model; }
} /// <summary>
/// 初始化宝马汽车
/// </summary>
public BwmCar() { }
/// <summary>
/// 初始化宝马汽车(带简单参数)
/// </summary>
/// <param name="model"></param>
public BwmCar(string model)
{
this.model = model;
}
/// <summary>
/// 启动宝马汽车
/// </summary>
/// <returns>是否启动成功</returns>
public bool Start()
{
Console.WriteLine("【{0} {1}】通过按键启动!", Name, Model);
return true;
}
}

  有了汽车产品标准,也有了两个具体的汽车产品类型,生产商还没有,生产标准也还没有!回顾下工厂方法模式的定义:定义一个用于创建对象的接口,但让子类决定要实例化哪个类!这里的“用于创建对象的接口”其实就是生产标准,而“子类”就是实现了生产标准的具体生产商。在我们的例子中就是汽车生产者接口(标准)和实现了该标准的具体汽车生产商:

    /// <summary>
/// 汽车生产接口(标准、规范)
/// </summary>
public interface ICarCreator
{
/// <summary>
/// 工厂级别的汽车生产接口方法(工厂方法模式的精髓所在!)
/// </summary>
/// <returns>生产好的汽车</returns>
ICar ProduceCar();
} /// <summary>
/// 奔驰生产者
/// </summary>
public class BenzCreator : ICarCreator
{
/// <summary>
/// 奔驰生产方法(对生产过程只做了简单模拟)
/// </summary>
/// <returns>奔驰汽车</returns>
public ICar ProduceCar()
{
ICar car = new BenzCar("梅赛德斯-AMG GLC 43 4MATIC");
Console.WriteLine("【Benz 梅赛德斯-AMG GLC 43 4MATIC】生产开始:\r1、生产发动机;\r2、生产底盘;\r3、生产车身;\r4、生产变速箱;\r5、生产轮胎;\r6、组装;");
Console.WriteLine("【Benz 梅赛德斯-AMG GLC 43 4MATIC】生产完成!");
return car;
}
} /// <summary>
/// 宝马生产者
/// </summary>
public class BwmCreator : ICarCreator
{
/// <summary>
/// 宝马生产方法(对生产过程只做了简单模似)
/// </summary>
/// <returns>宝马汽车</returns>
public ICar ProduceCar()
{
ICar car = new BwmCar("X7");
Console.WriteLine("【BWM X7】生产开始:\r1、生产底盘;\r2、生产发动机;\r3、生产变速箱;\r4、生产车身;\r5、生产轮胎;\r6、组装;");
Console.WriteLine("【BWM X7】生产完成!");
return car;
}
}

  我们可以看到,对于汽车生产标准,实现了该标准的不同汽车生产者可以自行决定生产什么类型的汽车产品以及如何生产——让子类决定要实例化哪个类!

  到目前为止,工厂方法(FactoryMethod)模式中各个角色都已经出现:用于创建对象的接口——汽车生产者接口(ICarCreator);子类(即具体的创建者)——奔驰生产者(BenzCreator)和宝马生产者(BwmCreator);要实例化的类(产品对象)——遵循统一产品标准(ICar)的奔驰汽车(BenzCar)和宝马汽车(BwmCar)。

  接下来我们来测试一下这样的对象创建构造符不符合工厂方法(FactoryMethod)模式的定义:定义一个用于创建对象的接口,但让子类决定要实例化哪个类。

    [TestClass]
public class FactoryMethodTest
{
[TestMethod]
public void TestFactoryMethod()
{
ICarCreator carCreator = new BenzCreator();//先将汽车创建者指向奔驰(Benz)汽车的创建者
ICar car = carCreator.ProduceCar();//由奔驰(Benz)汽车创建者决定生产什么车(实例化哪个类)
Assert.IsTrue(car.Start()); Console.Write("\r\r"); carCreator = new BwmCreator();//再将汽车创建者指向宝马(BWM)汽车的创建者
car = carCreator.ProduceCar();//再由宝马(BWM)汽车创建者决定生产什么车(实例化哪个类)
Assert.IsTrue(car.Start());
}
}

  可以看到客户代码已经可以让子类决定要实例化哪个类。我们最后来看看测试输出结果:

  最后,再次注意一下工厂方法(FactoryMethod)模式和构建者(Builder)模式的区别:

  1. 对要生产的产品制定了产品标准(IProduct),构建者模式中这个并不是必要的,当然也可以制定;
  2. 产品创建者标准(ICreator)只提供一个粗粒度的标准产品生产接口(FactoryMethod);
  3. 没有了额外的生产指导者,产品的具体创建流程和细节由具体的产品创建者(ConcreteCreator)自己决定;

  工厂方法(FactoryMethod)模式的使用场景在于模式定义的后半句:让子类决定要实例化哪个类(子类是指创建者子类)。也就是当你在开发过程中,对于创建对象有这样的需求时可以考虑一下工厂方法模式!而构建者模式主要使用场景在于用相同的构建过程构建复杂对象的不同表示。

设计模式之工厂方法(FactoryMethod)模式的更多相关文章

  1. 一天一个设计模式——工厂方法(FactoryMethod)模式

    一.模式说明 在前一个模板方法(Template Method)模式中,父类定义了处理流程,而流程中用到的方法交给子类去实现.类似的,在工厂方法模式中,父类决定如何生成实例,但并不决定所要生成的具体类 ...

  2. 【设计模式】——工厂方法FactoryMethod

    前言:[模式总览]——————————by xingoo 模式意图 工厂方法在MVC中应用的很广泛. 工厂方法意在分离产品与创建的两个层次,使用户在一个工厂池中可以选择自己想要使用的产品,而忽略其创建 ...

  3. 工厂方法(FactoryMethod)模式

    之前说了简单工厂设计模式如果增加一个新的运算的时候需要:增加一个具体的实现类,工厂类中增加一个case分支.也就是说我们不但对扩展开发了,也对修改开放了,违背了开闭原则.当然如果工厂类采用反射的话不存 ...

  4. 乐在其中设计模式(C#) - 工厂方法模式(Factory Method Pattern)

    原文:乐在其中设计模式(C#) - 工厂方法模式(Factory Method Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 工厂方法模式(Factory Method Pa ...

  5. 设计模式——从工厂方法模式到 IOC/DI思想

    回顾简单工厂 回顾:从接口的角度去理解简单工厂模式 前面说到了简单工厂的本质是选择实现,说白了是由一个专门的类去负责生产我们所需要的对象,从而将对象的创建从代码中剥离出来,实现松耦合.我们来看一个例子 ...

  6. C#设计模式(3)——工厂方法模式

    一.概念:定义一个用于创建对象的接口,让子类决定实例化哪一个类,工厂方法使一个类的实例化延迟到其子类. 二.代码实现 namespace 设计模式之工厂方法模式 { /// <summary&g ...

  7. C#设计模式(3)——工厂方法模式(转)

    C#设计模式(3)——工厂方法模式   一.引言 在简单工厂模式中讲到简单工厂模式的缺点,有一点是——简单工厂模式系统难以扩展,一旦添加新产品就不得不修改简单工厂方法,这样就会造成简单工厂的实现逻辑过 ...

  8. Java设计模式之工厂方法模式(转) 实现是抽象工厂?

    Java设计模式之工厂方法模式 责任编辑:覃里作者:Java研究组织   2009-02-25   来源:IT168网站   文本Tag: 设计模式 Java [IT168 技术文章]         ...

  9. php设计模式之工厂方法模式

    php设计模式之工厂方法模式 工厂方法模式 工厂方法模式(Factory Method Pattern)又称为工厂模式,也叫虚拟构造器(Virtual Constructor)模式或者多态工厂(Pol ...

随机推荐

  1. C#实时检测端口占用情况

    在TCP/IP协议中,服务端需要去监听客户端的端口,开始监听,我们需要检测使用的端口是否被占用,获取系统当前使用的所有端口号,用此端口进行匹配即可. 代码如下 internal static Bool ...

  2. 10_python_函数进阶

    一.函数参数-动态参数 形参:位置参数.默认值参数.动态参数 动态参数分为两种:动态接收位置参数 *args  .动态接收关键字参数 *kwargs     1. *args def chi(*foo ...

  3. 复习 C++ 中类的函数指针

    函数指针这种东西,平时工作中基本上不会用到. 那函数指针会用在哪里? 下面是一些基本的用法,根据消息号调到对应的函数: #include <iostream> #include <m ...

  4. apache ab

    ab -p postfile.json -T 'application/json' -n 100 -c 10 -v 2 http://192.168.1.103:3002/checkStashSlot ...

  5. spring boot实现异步调用

    今天在这里学习下使用springboot的异步调用async 首先使用@EnableAsync开启异步功能 /** * @author fengzp * @date 17/5/8 * @email f ...

  6. NPM(Node Package Manager,Node包管理器)

    简介 每个Node应用都有一个包含该应用元数据的文件-package.json,包含应用名.版本号以及依赖等信息. 我们使用NPM从NPM库下载并安装第三方包. 所有下载的包以及其依赖都保存在node ...

  7. Linux的mv 命令

    mv 命令的10个例子 1.移动文件 移动文件时需要注意的是文件的源地址和目标地址必须不同.这里有个例子,想要将file_1.txt文件从当前目录移动到其它目录,以/home/pungki/为例,语法 ...

  8. Scala使用隐式转换进行比较

    Boy.scala class Boy(val name: String, val faceValue: Int) extends Comparable[Boy]{ override def comp ...

  9. C# 多线程九之Timer类

    1.简介 相信写过定时任务的小伙伴都知道这个类,非常的轻量级,而且FCL中大量的类使用了这个方法,比如CancellationTokenSource的CancelAfter就是用Timer去做的. 当 ...

  10. 常见数据结构的Java实现

    单链表的Java实现 首先参考wiki上的单链表说明,单链表每个节点包含数据和指向链表中下一个节点的指针或引用.然后看代码 import java.lang.*; public class Singl ...