Abstract Factory Pattern(抽象工厂模式)

抽象工厂模式(Abstract Factory Pattern):抽象工厂的工作是将“抽象零件”组装成“抽象产品”。不关心零件的具体实现,而是值关心接口。我们使用该接口将零件组装成产品
在抽象工厂模式中,接口是负责创建一个相关对象的工厂,不需要显式指定它们的类。每个生成的工厂都能按照工厂模式提供对象。

介绍

  • 意图:提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。
  • 主要解决:主要解决接口选择的问题。
  • 何时使用:系统的产品有多于一个的产品族,而系统只消费其中某一族的产品。
  • 如何解决:在一个产品族里面,定义多个产品。
  • 关键代码:在一个工厂里聚合多个同类产品。
  • 优点:当一个产品族中的多个对象被设计成一起工作时,它能保证客户端始终只使用同一个产品族中的对象。
  • 缺点:产品族扩展非常困难,要增加一个系列的某一产品,既要在抽象的 Creator 里加代码,又要在具体的里面加代码。
  • 使用场景:
    1、QQ 换皮肤,一整套一起换。
    2、生成不同操作系统的程序。

注意事项:产品族难扩展,产品等级易扩展。

示例

Shape.ts 创建一个形状接口

export interface Shape {
    draw(): void;
}

Square.ts 正方形类实现形状接口

import { Shape } from "./Shape";

export default class Square implements Shape {
    public draw(): void {
        console.log("Inside Square::draw() method.");
    }
}

Rectangle.ts 长方形类实现形状接口

import { Shape } from "./Shape";

export default class Rectangle implements Shape {

    public draw(): void {
        console.log("Inside Rectangle::draw() method.");
    }
}

Circle.ts 圆形类实现形状接口

import { Shape } from "./Shape";

export default class Circle implements Shape {
    public draw(): void {
        console.log("Inside Circle::draw() method.");
    }
}

Color.ts 创建一个颜色接口。

export interface Color {
    fill(): void;
}

Red.ts 红色类实现颜色接口

import { Color } from "./Color";

export default class Red implements Color {

    public fill(): void {
        console.log("Inside Red::fill() method.");
    }
}

Blue.ts 蓝色类实现颜色接口

import { Color } from "./Color";

export default class Blue implements Color {

    public fill(): void {
        console.log("Inside Blue::fill() method.");
    }
}

Green.ts 绿色类实现颜色接口

import { Color } from "./Color";

export default class Green implements Color {

    public fill(): void {
        console.log("Inside Green::fill() method.");
    }
}

AbstractFactory.ts 抽象工厂,为Color、Shape创建抽象对象

import { Color } from "./Color";
import { Shape } from "./Shape";

export default abstract class AbstractFactory {
    public abstract getColor(color: string): Color;
    public abstract getShape(shape: string): Shape;
}

ColorFactory.ts 创建扩展了 AbstractFactory 的工厂类,基于给定的信息生成实体类的对象

import AbstractFactory from "./AbstractFactory";
import { Shape } from "./Shape";
import { Color } from "./Color";
import Red from "./Red";
import Green from "./Green";
import Blue from "./Blue";

export default class ColorFactory extends AbstractFactory {
    public getShape(shapeType: string): Shape {
        return null;
    }
    public getColor(color: string): Color {
        if (color == null) {
            return null;
        }
        if (color.toUpperCase() === "RED") {
            return new Red();
        } else if (color.toUpperCase() === "GREEN") {
            return new Green();
        } else if (color.toUpperCase() === "BLUE") {
            return new Blue();
        }
        return null;
    }
}

ShapeFactory.ts 创建扩展了 AbstractFactory 的工厂类,基于给定的信息生成实体类的对象

import AbstractFactory from "./AbstractFactory";
import { Shape } from "./Shape";
import Circle from "./Circle";
import Rectangle from "./Rectangle";
import Square from "./Square";
import { Color } from "./Color";

export default class ShapeFactory extends AbstractFactory {

    public getShape(shapeType: string): Shape {
        if (shapeType == null) {
            return null;
        }
        if (shapeType.toUpperCase() === "CIRCLE") {
            return new Circle();
        } else if (shapeType.toUpperCase() === "RECTANGLE") {
            return new Rectangle();
        } else if (shapeType.toUpperCase() === "SQUARE") {
            return new Square();
        }
        return null;
    }
    public getColor(color: string): Color {
        return null;
    }
}

FactoryProducer.ts 创建一个工厂创造器/生成器类,通过传递形状或颜色信息来获取工厂

import AbstractFactory from "./AbstractFactory";
import ShapeFactory from "./ShapeFactory";
import ColorFactory from "./ColorFactory";

export default class FactoryProducer {
    public static getFactory(choice: string): AbstractFactory {
        if (choice.toUpperCase() === "SHAPE") {
            return new ShapeFactory();
        } else if (choice.toUpperCase() === "COLOR") {
            return new ColorFactory();
        }
        return null;
    }
}

index.ts

import FactoryProducer from "./FactoryProducer";
//获取形状工厂
const shapeFactory = FactoryProducer.getFactory("SHAPE");

//获取形状为 Circle 的对象
const shape1 = shapeFactory.getShape("CIRCLE");

//调用 Circle 的 draw 方法
shape1.draw();

//获取形状为 Rectangle 的对象
const shape2 = shapeFactory.getShape("RECTANGLE");

//调用 Rectangle 的 draw 方法
shape2.draw();

//获取形状为 Square 的对象
const shape3 = shapeFactory.getShape("SQUARE");

//调用 Square 的 draw 方法
shape3.draw();
//获取颜色工厂
const colorFactory = FactoryProducer.getFactory("COLOR");

//获取颜色为 Red 的对象
const color1 = colorFactory.getColor("RED");

//调用 Red 的 fill 方法
color1.fill();

//获取颜色为 Green 的对象
const color2 = colorFactory.getColor("Green");

//调用 Green 的 fill 方法
color2.fill();

//获取颜色为 Blue 的对象
const color3 = colorFactory.getColor("BLUE");

//调用 Blue 的 fill 方法
color3.fill();

result

Inside Circle::draw() method.
Inside Rectangle::draw() method.
Inside Square::draw() method.
Inside Red::fill() method.
Inside Green::fill() method.
Inside Blue::fill() method.

类图

角色

  • AbstractProduct(抽象产品)
    AbstractProduct角色负责定义AbstractFactory角色所生产的抽象零件和产品的接口
  • AbstractFactory(抽象工厂)
    AbstractFactory角色负责定义用于生产抽象产品的接口
  • Client(委托者)
    Client角色仅会调用AbstractFactory角色和AbstractProduct角色的接口进行工作,对于具体的零件、产品和工厂一无所知。
  • ConcreteProduct(具体产品)
    ConcreteProduct角色负责实现AbstractProduct角色定义的接口
  • ConcreteFactory(具体工厂)
    ConcreteFactory角色负责实现AbstractFactory角色的接口。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容