http://blog.csdn.net/chollima/article/details/3985570
一.简单工厂模式又称静态工厂方法模式(Static Factory Method),它不是Gof 所讲的23种设计模式之一,但是它却是我们在编码过程中经常使用的方法之一。
1.静态工厂方法统一管理对象的创建。
静态工厂方法通过传入的参数判断决定创建哪一个产品的实例,封装了对象的创建,客户端只管消费,实现了对责任(模块)的分割。
2.静态工厂方法推迟了产品的实例化。
通过XML配置文件就能改变具体要创建的产品实例,修改为其它的产品实例,代码不须重新编译。
二.简单工厂模式还是有缺点的,后面的工厂方法模式和抽象工厂模式就是对这些缺点的改善。讲完了这三种模式将会有一个对比。以下以Nokia手机为例,采用简单工厂模式设计的源代码雏形。
- /* Nokia抽象产品 */
- #pragma once
- #include <string>
- #include <iostream>
- class CNokia
- {
- public:
- CNokia(void);
- public:
- virtual ~CNokia(void);
- public:
- virtual bool MakeCall(const std::string & number) const = 0;
- };
- #include "Nokia.h"
- CNokia::CNokia(void)
- {
- }
- CNokia::~CNokia(void)
- {
- }
- #pragma once
- #include "nokia.h"
- class CN96 : public CNokia
- {
- public:
- CN96(void);
- public:
- virtual ~CN96(void);
- public:
- virtual bool MakeCall(const std::string & number) const;
- };
- /* N96型号的Nokia手机产品类 */
- #include "N96.h"
- CN96::CN96(void)
- {
- std::cout << "生产一部N96" << std::endl;
- }
- CN96::~CN96(void)
- {
- }
- bool CN96::MakeCall(const std::string & number) const
- {
- std::cout << "我在用N96,拨打号码是:" << number.c_str() << std::endl;
- return false;
- }
- #pragma once
- #include "nokia.h"
- class CN95 :public CNokia
- {
- public:
- CN95(void);
- public:
- virtual ~CN95(void);
- public:
- virtual bool MakeCall(const std::string & number) const;
- };
- /* N95型号的Nokia手机产品类 */
- #include "N95.h"
- CN95::CN95(void)
- {
- std::cout << "生产一部N95" << std::endl;
- }
- CN95::~CN95(void)
- {
- }
- bool CN95::MakeCall(const std::string & number) const
- {
- std::cout << "我在用N95,拨打号码是:" << number.c_str() << std::endl;
- return false;
- }
- /*N85, N81, ...*/
- /* 工厂类 */
- #pragma once
- #include "N96.h"
- #include "N95.h"
- #include "N85.h"
- #include "N81.h"
- #include <cassert>
- class CNokiaSimpleFactory
- {
- public:
- CNokiaSimpleFactory(void);
- public:
- ~CNokiaSimpleFactory(void);
- public:
- /* 静态工厂方法 */
- static CNokia * CreateNokia(const std::string & model);
- };
- #include "NokiaSimpleFactory.h"
- CNokiaSimpleFactory::CNokiaSimpleFactory(void)
- {
- }
- CNokiaSimpleFactory::~CNokiaSimpleFactory(void)
- {
- }
- CNokia * CNokiaSimpleFactory::CreateNokia(const std::string & model)
- {
- if (model == "N96")
- {
- return new CN96();
- }
- else if (model == "N95")
- {
- return new CN95();
- }
- else if (model == "N85")
- {
- return new CN85();
- }
- else if (model == "N81")
- {
- return new CN81();
- }
- else
- {
- assert(false);
- }
- return NULL;
- }
- /* main方法,可以看做客户端 */
- #include "stdafx.h"
- #include "NokiaSimpleFactory.h"
- int _tmain(int argc, _TCHAR* argv[])
- {
- CNokia * nokia = NULL;
- /*
- modeName 可以从外部XML文件中读取,运行过程中动态的
- 决定该创建哪一种型号的手机
- */
- std::string modeName = "N96";
- nokia = CNokiaSimpleFactory::CreateNokia(modeName);
- nokia->MakeCall("123456789");
- delete nokia;
- nokia = NULL;
- return 0;
- }
工厂方法模式是对简单工厂模式的改进。首先看看简单工厂模式的缺点。
软件是之所以区别于程序,是因为它可以被人们使用,并能间接创造效益。需求是软件开发的核心,忽视用户的需求,软件本身就没有存在的价值。
假如Nokia又新开发了一款新手机N99,对于采用简单工厂模式设计的系统,我们的做法是:
1.新增一个CN99类继承自CNokia抽象类,这个符合OCP原则。
注:OCP:Open-Closed Principle开闭原则。OCP关注的是灵活性,改动是通过增加代码进行的,而不是改动现有的代码。对扩展开放,对修改关闭。
2.修改静态工厂方法中的判断逻辑,增加对N99的创建代码。这个明显违背OCP。
改进简单工厂模式,我们要做的就是克服2中的缺点。
下面的工厂方法模式,新增一个N96的工厂类,不用修改已有的代码,完全支持OCP原则,而且工厂方法模式也完全遵守里氏替换原则,LSP原则是OCP成为可能的重要原则。
注:Liskov Substitution Principle(里氏代换原则):子类型(subtype)必须能够替换它们的基类型。
虽然工厂方法模式是对简单工厂模式缺点的该进,这个缺点就是违背软件设计原则OCP。但是并不是工厂方法模式就优于简单工厂模式,只不过后者更符合常规的软件设计理念。
- /* 抽象工厂类 */
- #pragma once
- #include "Nokia.h"
- class CFactoryMethod
- {
- public:
- CFactoryMethod(void);
- public:
- virtual ~CFactoryMethod(void);
- public:
- virtual CNokia * CreateNokiaMobile(void) = 0;
- };
- #include "FactoryMethod.h"
- CFactoryMethod::CFactoryMethod(void)
- {
- }
- CFactoryMethod::~CFactoryMethod(void)
- {
- }
- /* 生产N99的具体工厂类 */
- #pragma once
- #include "factorymethod.h"
- #include "N96.h"
- class CN96Factory :
- public CFactoryMethod
- {
- public:
- CN96Factory(void);
- public:
- virtual ~CN96Factory(void);
- public:
- virtual CNokia * CreateNokiaMobile(void);
- };
- #include "N96Factory.h"
- CN96Factory::CN96Factory(void)
- {
- }
- CN96Factory::~CN96Factory(void)
- {
- }
- CNokia * CN96Factory::CreateNokiaMobile(void)
- {
- return new CN96();
- }
- /* 客户端通过建立N99工厂生产Nokia手机 */
- #include "stdafx.h"
- #include "N96Factory.h"
- int _tmain(int argc, _TCHAR* argv[])
- {
- CFactoryMethod * factoryMethod = new CN96Factory();
- CNokia * nokia = factoryMethod->CreateNokiaMobile();
- nokia->MakeCall("1234567");
- return 0;
- }
Factory Method使一个类的实例化延迟到其子类。