GOF:提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。

  类图:

  


  

  观察类图,关键就在于左边,即AbstractFactory和它的两个子类。想要理解这个模式,我们可以真的将它看作一个工厂。对于生产家电的工厂,一般要生产电视和洗衣机,但是在美国的家电工厂生产的是等离子电视和全自动洗衣机,在非洲的家电工厂生产的是CRT彩电和半自动洗衣机。把这个例子转换成类似上面的类图就如下:

  这时,如果想要开工厂,就能从彩电工厂生产标准知道彩电工厂能生产电视和洗衣机,然后根据你在哪个国家来具体决定彩电工厂的类型。

  AbstractFactory一般是抽象类(但不是绝对的),它定义了必需的接口,而具体接口的实现放到了子类。用户只需知道Factory中有什么接口就行了,剩下就交给多态吧(在设计模式中,很多巧妙的地方就是靠多态实现的)。

  比如:一群学生要去参观家电工厂的生产过程,则代码如下:

void StudentVisit(AbstractFactory& factory) {
    factory.MakeTV();
    factory.MakeWasher();
}

int main() {
    USAFactory factory;
    StudentVisit(factory);

    ;
}

  这里实际传入的参数是美国家电工厂,则实际调用的是美国家电工厂的MakeTV()和MakeWasher()。

  适用性:

  • 一个系统要独立于它的产品的创建,组合和表示时。(我们只需要知道有什么接口,而不需要了解子类是怎么实现的)
  • 一个系统要由多个产品系列中的一个来配置时。(比如上面电视有多种,但是美国彩电工厂要的是等离子电视)
  • 当你要强调一系列相关的产品对象的设计以便进行联合使用时。(彩电工厂需要生产电视和洗衣机,这两个就是相关的产品)
  • 当你提供一个产品类库,而只想显示它们的接口而不是实现时。(我们只需要调用抽象类的接口,而不需要到子类中看具体的方法)

  优点 + 缺点:

  • 它分离了具体的类。
  • 它使得易于交换产品系列。改变使用的具体工厂就能生产不同的产品。
  • 它有利于产品的一致性。每一个具体工厂生产的都是同一等级的产品。
  • 难以支持新种类的产品。如果要在抽象类中增加接口,每个具体工厂都要进行修改。

 


  最后以一个创建游戏场景的例子结束,该游戏场景会产生地形,光源,植被。

  类图:

  代码:

  先看看Product,即类图右边的类:

  地形类:

class Terrain {
public:
    ;
};

class DesertTerrain : public Terrain {
public:
    DesertTerrain() {
        m_type = "Desert";
    }
    string GetType() const {
        return m_type;
    }
private:
    string m_type;
};

class MountainTerrain : public Terrain {
public:
    MountainTerrain() {
        m_type = "Mountain";
    }
    string GetType() const {
        return m_type;
    }
private:
    string m_type;
};

  光源类:

class Light {
public:
    ;
};

class SpotLight : public Light{
public:
    SpotLight() {
        m_type = "Spot Light";
    }
    string GetType() const {
        return m_type;
    }
private:
    string m_type;
};

class DirectionLight : public Light {
public:
    DirectionLight() {
        m_type = "Direction Light";
    }
    string GetType() const {
        return m_type;
    }
private:
    string m_type;
};

  植被类:

class Plant {
public:
    ;
};

class TreePlant : public Plant {
public:
    TreePlant() {
        m_type = "Tree";
    }
    string GetType() const {
        return m_type;
    }
private:
    string m_type;
};

class CactusPlant : public Plant {
public:
    CactusPlant() {
        m_type = "Cactus";
    }
    string GetType() const {
        return m_type;
    }
private:
    string m_type;
};

  

 以下就看工厂类:

class GameSceneFactory {
public:

    ;
    ;
    ;
};

class DesertSceneFactory : public GameSceneFactory {
public:

    Terrain* MakeTerrain() const {
        return new DesertTerrain();
    }

    Light* MakeLight() const {
        return new DirectionLight();
    }

    Plant* MakePlant() const {
        return new CactusPlant();
    }
};

class MountainSceneFactory : public GameSceneFactory {
public:

    Terrain* MakeTerrain() const {
        return new MountainTerrain();
    }

    Light* MakeLight() const {
        return new SpotLight();
    }

    Plant* MakePlant() const {
        return new TreePlant();
    }
};

  以下就是具体的使用:

class Game {
public:
    void createScene(GameSceneFactory& factory) {
        m_terrain = factory.MakeTerrain();
        m_light = factory.MakeLight();
        m_plant = factory.MakePlant();
        print();
    }
    void print() {
        cout << "------Game Scene------" << endl;
        cout << "Terrain: " << m_terrain->GetType() << endl;
        cout << "Light: " << m_light->GetType() << endl;
        cout << "Plant: " << m_plant->GetType() << endl;
    }
private:
    Terrain* m_terrain;
    Light* m_light;
    Plant* m_plant;
};

int main() {
    Game game;
    MountainSceneFactory factory;
    game.createScene(factory);
}

  结果:


  小结:抽象工厂模式是一个比较常用的模式,需要好好掌握。利用真实生活中的工厂就比较好理解该模式的运作,但真正要掌握还得在日常编码中的适当场合使用。

  

抽象工厂模式(Abstract Factory)的更多相关文章

  1. .NET设计模式(2):1.2 抽象工厂模式(Abstract Factory)

    概述 抽象工厂模式(Abstract Factory)是所有形态的工厂模式中最为抽象和最具一般性的一种形态.抽象工厂模式是指当有多个抽象角色时,使用的一种工厂模式.抽象工厂模式可以向客户端提供一个接口 ...

  2. 乐在其中设计模式(C#) - 抽象工厂模式(Abstract Factory Pattern)

    原文:乐在其中设计模式(C#) - 抽象工厂模式(Abstract Factory Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 抽象工厂模式(Abstract Factor ...

  3. 【设计模式】抽象工厂模式 Abstract Factory Pattern

    简单工厂模式是一个工厂类根据工厂方法的参数创建不出不同的产品, 工厂方法模式是每一个产品都有一个一一对应的工厂负责创建该产品.那么今天要讲的抽象工厂模式是一个工厂能够产生关联的一系列产品.抽象工厂模式 ...

  4. 设计模式 - 抽象工厂模式(abstract factory pattern) 具体解释

    抽象工厂模式(abstract factory pattern) 详细解释 本文地址: http://blog.csdn.net/caroline_wendy/article/details/2709 ...

  5. 抽象工厂模式(Abstract Factory)C#实例

    抽象工厂模式(Abstract Factory)C#实例 本文出处http://www.dofactory.com/net/abstract-factory-design-pattern 一.场景描述 ...

  6. 二十四种设计模式:抽象工厂模式(Abstract Factory Pattern)

    抽象工厂模式(Abstract Factory Pattern) 介绍提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类. 示例有Message和MessageModel,Messag ...

  7. 【UE4 设计模式】抽象工厂模式 Abstract Factory Pattern

    概述 描述 提供一个创建一系列相关或相互依赖对象的接口,而无须指定它们具体的类:具体的工厂负责实现具体的产品实例 抽象工厂中每个工厂可以创建多种产品(如苹果公司生产iPhone.iPad): 工厂方法 ...

  8. Net设计模式实例之抽象工厂模式(Abstract Factory Pattern)

    一.抽象工厂模式简介(Bref Introduction) 抽象工厂模式(Abstract Factory Pattern),提供一个创建一系列相关或者相互依赖对象的接口,而无需制定他们的具体类.优点 ...

  9. C#设计模式——抽象工厂模式(Abstract Factory Pattern)

    一.概述在软件开发中,常常会需要创建一系列相互依赖的对象,同时,由于需求的变化,往往存在较多系列对象的创建工作.如果采用常规的创建方法(new),会造成客户程序和对象创建工作的紧耦合.对此,抽象工厂模 ...

  10. Objective-C设计模式——抽象工厂模式Abstract Factory(对象创建)

    抽象工厂模式 理解了工厂方法模式,其实抽象工厂和工厂方法模式有很多的相似之处.抽象工厂同样是分离客户端对象的创建和逻辑代码的,但是抽象工厂往往是产生一组数据而不单单是产生一个产品. 抽象工厂提供一个创 ...

随机推荐

  1. windows安装python

    1:首先去python网站下载安装包:https://www.python.org/downloads/,注意自己的系统版本 2:在自己指定目录安装即可: 3:将python路径加入PATH环境变量: ...

  2. java,我准备好了

    1.你对自己的未来有什么规划?做了哪些准备? 长期上学好公务员和本专业知识,将来能找一份好工作,并能在职务上履行好相称的工作.短期上在下一学期拿一等奖学金,尽快入党,考出英语四级和二级c语言.任何成就 ...

  3. Dijkstra(变形) POJ 1797 Heavy Transportation

    题目传送门 题意:求1到n的最大载重量 分析:那么就是最大路上的最小的边权值,改变优先规则. #include <cstdio> #include <algorithm> #i ...

  4. HDU1402 A * B Problem Plus(FFT)

    http://acm.hdu.edu.cn/showproblem.php?pid=1402 初学FFT. http://www.cnblogs.com/WABoss/p/FFT_Note.html ...

  5. 【原】iOS学习41之多线程

    1. 多线程概述 1> 程序.进程和进程的概念 程序:由源代码生成的可执行应用.(例如:QQ.app) 进程:一个正在运行的程序可以看做一个进程.(例如:正在运行的QQ就是一个进程),进程拥有独 ...

  6. BZOJ4157 : 星际瘟疫

    首先剔除所有从$R$不可到达的点,然后用Lengauer-Tarjan算法建立出以$R$为起点的Dominator Tree. 那么对于每个询问,求出那些点的父亲的LCA,那么答案就是LCA到根路径上 ...

  7. 【CodeVS】p1299 切水果

    题目描述 Description 简单的说,一共N个水果排成一排,切M次,每次切[L,R]区间的所有水果(可能有的水果被重复切),每切完一次输出剩下水果数量 数据已重新装配,不会出现OLE错误 时限和 ...

  8. Idea_Intellij Idea 12 生成serialVersionUID的方法

    默认情况下Intellij IDEA是关闭了继承了java.io.Serializable的类生成serialVersionUID的警告.如果需要ide提示生成serialVersionUID,那么需 ...

  9. 用c#实现$.now()(1437813924915)的时间效果

    Convert.ToInt64((DateTime.UtcNow - Convert.ToDateTime("1970-01-01")).TotalMilliseconds)

  10. 让Xcode的 stack trace信息可读

    让Xcode的 stack trace信息可读 昨天在写 iOS 代码的时候,调试的时候模拟器崩溃了.异常停在了如下整个 main 函数的入口处: int main(int argc, char *a ...