Pattern Tips 之三

Pattern Tips 之三
作者:温昱

感谢:《设计模式》一书的作者Gamma,Helm,Johnson和Vilssides,译者李英军等

----------------------------------说明----------------------------

Abstract Factory,Factory Method,Prototype。它们都是Creationall Patterns,它们的关系如下图所示:



----------------------------------Abstract Factory----------------------------

●Tip 1:关键字。Family。

●Tip 2:图。书中的原图如下,



我又画了一张,主要是为我后边讨论本模式“支持变化”做准备,



可以看到,Client用了一个Factory,一个Factory可以“生产”一个Product Family(就是多个Product或A Set of Products),Client使用这一个Product Family。

●Tip 3:实现和使用。

从图中可以看到,ConcreteFactory负责实例化ConcreteProduct,但谁来实例化ConcreteFactory呢?图中Client Class仅对AbstractFactory有关联,但是一个Client Class是不能拥有一个Interface的实例的,而只能拥有它的指针或引用(来支持多态);那么Client Class能new AbstractFactory吗,不仅不能,而且要new ConcreteFactory才对呀。

结论,当“Architect或Achitecture工程师”“实现”Abstract Factory模式时,ConcreteProduct的实例化问题已经考虑在内了,因为这恰恰就是该模式的使命。而当“Application工程师”“使用”Abstract Factory模式时,应负责实例化ConcreteFactory。

在ET++中,Abstract Factory模式实现得非常巧妙。具体实例化哪个ConcreteFactory是“Architect或Achitecture工程师”写在Framework中的一些If语句决定的,这些If语句去读环境变量,解除了“Application工程师”去实例化ConcreteFactory的职责。ET++中的WindowSystem的相关研究,请参考本站(lcspace.nease.net)的Framework栏目。

在MFC中,CDocTemplate也是Abstract Factory模式,并且也非常巧妙地用CRumtimeClass机构解决具体实例化哪个ConcreteFactory的问题。

●Tip 4:支持变化。It makes exchanging product families easy。

不同Product Family一般用于支持类似“多种平台”这样的情形(所以Client一般不同时使用多个Product Family)。想像“Application工程师”欲将Client移植到新平台,由于Client Class对AbstractProduct和AbstractFactory是“良性依赖”(在图中用绿色标出),所以不会引起Client Class的代码变动。如果Client Class还负责“实例化ConcreteProduct”,那么只需改区区2处,比如从

#include "BombedMazeFacory.h"
BombedMazeFacory factory;
game.CreateMaze( factory );
改变成
#include "AnotherMazeFacory.h"  //changed
AnotherMazeFacory factory;  //changed
game.CreateMaze( factory );
增加新的Product Family,对“Architect或Achitecture工程师”来说,只需新写图中黄色Class:新写了一个New ConcreteFactory以及它生产的New Product Family。

在MFC中,CFrame/CDocument/CView就是一个Product Family,我们的确很容易地创建New Product Fameliy:CMyFrame/CMyDoc/CMyView。

●Tip 5:局限性。Supporting new kinds of products is difficult。

简单说,就是将“支持这6个Product”换成“支持那6个Product”容易,换成“支持那7个Product”难。因为AbstractFactory的操作的个数和Product Family中Product的个数相对应,它限制了所有ConcreteFactory的接口,这个接口不应也不易随便改。在图中表现得很充份,Client可以有多个,AbstractProduct可以有多个,Product Family可以有多个,ConcreteFactory可以有多个,唯独AbstractFactory只有一个。

在MFC中,CFrame/CDocument/CView就是一个Product Family,你想在Doc/View arch下创建一个4员组,难。

----------------------------------Factory Method----------------------------

●Tip 1:关键字。Subclass,Virtual,Override。

●Tip 2:图。



●Tip 3:实现和使用。

A factory method define an interface for creating an object,它只要求该Method的责任是Factory:你可以将它实现成纯虚函数/空函数/做相关创建的函数,用于不同情况。但是,在C++中它总是虚函数。

在MFC中,CWnd::Create()就是典型的例子。

●Tip 4:支持变化。

可以看到,2个基类Superclass和AbstractProduct是Framework的组成部份,并且Superclass对AbstractProduct是“对接口编程”,所以是“良性依赖”(在图中用绿色标出),不易引起Superclass的代码变动。

图中黄色的类是New Application新加的,Subclass2对Product2是“恶性依赖”(在图中用红色标出),但又有什么关系呢?因为它们都属于Application,而且Subclass2::FactoryMethod()仅仅创建Product2,涉及的代码很少。

●Tip 5:支持框架。Factory methods pervade toolkits and frameworks。The framework must instantiate classes, but it only knows about abstract classes, which it cannot instantiate。Factory methods eliminate the need to bind application-specific classes into your code。A factory method define an interface for creating an object, but let subclasses CAN (may not) decide which class to instantiate。

----------------------------------Prototype----------------------------

●Tip 1:关键字。Copy,Clone itself。

●Tip 2:图。



●Tip 3:实现和使用。

Wall * BombedWall::Clone() const
{
    return new BombedWall( *this );
}
在MFC中,CWnd::Create()就是典型的例子。

●Tip 4:优点。

仅有 1 个“类层次”,而不是象Factory Method一样平行的 2 个。

------------------Abstract Factory,Factory Method,Prototype,Template Method-----------------

●Tip 1:下面是2种典型的应用情况(其中使用Abstract Factory的情况又分2种实现):



可以看到,Abstract Factory模式常要使用Factroy Method模式或者Prototype模式,Template Method经常调用Factroy Method,



●Tip 2:Abstract Factory模式使用Factroy Method模式之例──使用Abstract Factory的MazeGame。注意MazeFactory已内置缺省实现故不是AbstractMazeFactory;还要注意哪里是“对接口编程”,哪里是“对子类名硬编码”。
/////////////////////////////////////////////////////AbstractFactory
#include "Maze.h"
#include "Room.h"
#include "Wall.h"

class MazeFactory //a superclass as a AbstractFactory, but not containing pure virtual method
{
public:
  MazeFactory(); //con
  virtual Maze* MakeMaze() //a factory method, which must be virtual
    {return new Maze;}
  virtual Room * MakeRoom(int n) //a factory method, which must be virtual
    {return new Room(n);}
  virtrul Wall * MakeWall() //a factory method, which must be virtual
    {return new Wall;}
};
/////////////////////////////////////////////////////Client
#include "BombedMazeFacory.h" //这里是对子类名硬编码(类名和文件名一样)

void MazeGame::XXXXXXXX ( void )
{
  BombedMazeFacory factory;  //实例化ConcreteFactory,这里是对子类名硬编码
  CreateMaze( factory );
}

Maze* MazeGame::CreateMaze (MazeFactory & factory) //BombedMazeFacory.h must
                         //   have included MazeFactory.h, so MazeFactory is available
{
  //
  //these 4 statements call Factory to create some Products
  //这里是对接口编程
  //
  Maze* aMaze = factory.MakeMaze();
  Room* r1 = factory.MakeRoom(1);
  Room* r2 = factory.MakeRoom(2);
  Door* aDoor = factory.MakeDoor(r1, r2);
  
  //
  //all the rest statements use these Products
  //这里是对接口编程
  //
  aMaze->AddRoom(r1); 
  aMaze->AddRoom(r2);
  
  r1->SetSide(North, factory.MakeWall());
  r1->SetSide(East, aDoor);
  r1->SetSide(South, factory.MakeWall());
  r1->SetSide(West, factory.MakeWall());
  
  r2->SetSide(North, factory.MakeWall());
  r2->SetSide(East, factory.MakeWall());
  r2->SetSide(South, factory.MakeWall());
  r2->SetSide(West, aDoor);
  
  return aMaze;
}
/////////////////////////////////////////////////////ConcreteFactory
#include "MazeFactory.h"
#include "BombedMaze.h"

class BombedMazeFactory : public MazeFactory //a subclass as a ConcreteFactory
{
public:
  BombedMazeFactory();
  virtual Maze* MakeMaze() //override a factory method, which must be virtual
    {return new BombedMaze;}
};
●Tip 3:Template Method经常调用Factroy Method之例──使用Template Method的MazeGame。注意与使用Abstract Factory的MazeGame比较。
/////////////////////////////////////////////////////Superclass
class MazeGame 
{
public:
  MazeGame(); //con
  
  Maze* CreateMaze(); //a Template Method
  
  virtual Maze* MakeMaze();//4 factory methods, which must be virtual 
  virtual Room* MakeRoom(int n);
  virtual Wall* MakeWall() const
  virtual Door* MakeDoor(Room* r1, Room* r2);
};
/////////////////////////////////////////////////////Template Method
Maze* MazeGame::CreateMaze () 
{
  //
  //本模板方法调用工厂方法MakeXxx()
  //
  
  Maze* aMaze = MakeMaze();
  
  Room* r1 = MakeRoom(1);
  Room* r2 = MakeRoom(2);
  Door* theDoor = MakeDoor(r1, r2);
  
  aMaze->AddRoom(r1);
  aMaze->AddRoom(r2);
  
  r1->SetSide(North, MakeWall());
  r1->SetSide(East, theDoor);
  r1->SetSide(South, MakeWall());
  r1->SetSide(West, MakeWall());
  
  r2->SetSide(North, MakeWall());
  r2->SetSide(East, MakeWall());
  r2->SetSide(South, MakeWall());
  r2->SetSide(West, theDoor);
  
  return aMaze;
}
/////////////////////////////////////////////////////Subclass
#include "MazeGame.h"
#include "BombedWall.h"
#include "RoomWithABomb.h"

class BombedMazeGame : public MazeGame 
{
public:
  BombedMazeGame(); //con
  
  virtual Wall* MakeWall() //override a factory method, which must be virtual
  { return new BombedWall; }
  
  virtual Room* MakeRoom(int n) //override a factory method, which must be virtual
  { return new RoomWithABomb(n); }
};
posted @ 2004-07-27 09:49  R2  阅读(726)  评论(0编辑  收藏  举报