抽象工厂模式 Abstract Factory Pattern

一、模式介绍

工厂方法模式考虑的是一类产品的生产,如电视机厂只生成电视机、披萨店只生产披萨等。

同类产品称为同等级,也就是说:工厂方法模式只考虑生产同等级的产品,但是在现实生活中许多工厂是综合型的工厂,能生产多等级(种类)的产品。如农场里既养动物又种植物,肯德基既生产汉堡有生产炸鸡等。

抽象工厂模式是工厂方法模式的升级版本,工厂方法模式只生产一个等级的产品,而抽象工厂模式可生产多个等级的产品。将同一个具体工厂所生产的位于不同等级的一组产品称为一个产品族,如下图所示的海尔工厂和 TCL 工厂所生成的电视机和空调的对应关系。
在这里插入图片描述

1.1、定义

抽象工厂模式提供一个接口,用于创建相关或相互依赖对象的家族,而不需要指定所要产品的具体类。

使用抽象工厂模式一般要满足以下条件:

  1. 系统中多个产品族,每个具体工厂创建同一族但属于不同等级结构的产品
  2. 系统一次只克而可能消费其中某一族的产品,即同族产品一起使用。

1.2、优点

  1. 具有工厂模式的优点
  2. 可以在类的内部对产品族中相关联的多等级产品共同管理,而不必专门引入多个新的类来进行管理
  3. 当需要产品族时,抽象工厂可以保证客户端始终只是用同一个产品的产品族
  4. 抽象工厂增强了程序的可扩展性,当增加一个新的产品族时,不需要修改源代码,满足开闭原则

1.3、缺点

  1. 当产品族中需要增加一个新的产品时,所有的工厂类都需要进行修改
  2. 增加了系统的抽象性和理解难度

二、结构与实现

2.1、结构

抽象工厂模式同工厂方法模式一样,也是由抽象工厂、具体工厂、抽象产品和具体产品等四个要素构成,但抽象工厂中方法个数不同,抽象产品的个数也不同。

  1. 抽象工厂(AbstractFactory):提供了创建产品的接口,它包含多个创建产品的方法 newProduct() ,可以创建多个不同等级的产品。
  2. 具体工厂(ConcreteFactory):主要是实现抽象工厂中的抽象方法,完成具体产品的创建。
  3. 抽象产品(Product):定义了产品的规范,描述了产品的主要特性和功能,抽象工厂有多个抽象产品。
  4. 具体产品(ConcreteProduct):实现了抽象产品角色所定义的接口,由具体工厂来创建,它同具体工厂之间一一对应。
    在这里插入图片描述

2.2、实现

2.2.1、类图

在这里插入图片描述

2.2.2、NYPizzaStore
package com.erlang.factory.abstract_;


/**
 * @description: 纽约披萨
 * @author: erlang
 * @since: 2022-02-08 22:28
 */
public class NYPizzaStore extends PizzaStore {

    @Override
    protected Pizza createPizza(String type) {
        Pizza pizza = null;
        PizzaIngredientFactory factory = new NYPizzaIngredientFactory();
        if (type.equals("cheese")) {
            pizza = new CheesePizza(factory);
            pizza.setName("NY Style Cheese Pizza");
        } else if (type.equals("clam")) {
            pizza = new ClamPizza(factory);
            pizza.setName("NY Style Clam Pizza");
        }
        return pizza;
    }
}
2.2.3、PizzaIngredientFactory
package com.erlang.factory.abstract_;

/**
 * @description: 芝加哥披萨成分加工工厂
 * @author: erlang
 * @since: 2022-02-09 22:38
 */
public interface PizzaIngredientFactory {
    /**
     * 面团
     *
     * @return 面团
     */
    Dough createDough();

    /**
     * 酱汁
     * @return
     */
    Sauce createSauce();
}
2.2.4、NYPizzaIngredientFactory
package com.erlang.factory.abstract_;

/**
 * @description: 纽约披萨
 * @author: erlang
 * @since: 2022-02-09 22:59
 */
public class NYPizzaIngredientFactory implements PizzaIngredientFactory {

    @Override
    public Dough createDough() {
        return new ThinCrustDough("Thin Crust Dough");
    }

    @Override
    public Sauce createSauce() {
        return new MarinaraSauce("Marinara Sauce");
    }
}
2.2.5、ChicagoPizzaIngredientFactory
package com.erlang.factory.abstract_;

/**
 * @description: 芝加哥披萨
 * @author: erlang
 * @since: 2022-02-09 22:39
 */
public class ChicagoPizzaIngredientFactory implements PizzaIngredientFactory {

    @Override
    public Dough createDough() {
        return new ThickCrustDough("Extra Thick Crust Dough");
    }

    @Override
    public Sauce createSauce() {
        return new PlumTomatoSauce("Plum Tomato Sauce");
    }
}
2.2.6、Dough
package com.erlang.factory.abstract_;

/**
 * @description: 面团接口
 * @author: yj
 * @since: 2022-02-09 22:42
 */
public interface Dough {

    /**
     * 面团
     *
     * @return 返回面团
     */
    String getDough();
}
2.2.7、ThickCrustDough
package com.erlang.factory.abstract_;

/**
 * @description:
 * @author: erlang
 * @since: 2022-02-09 22:53
 */
public class ThickCrustDough implements Dough {

    private String dough;

    public ThickCrustDough(String dough) {
        this.dough = dough;
    }

    @Override
    public String getDough() {
        return dough;
    }
}

2.2.8、ThinCrustDough
package com.erlang.factory.abstract_;

/**
 * @description:
 * @author: erlang
 * @since: 2022-02-09 22:57
 */
public class ThinCrustDough implements Dough {

    private String dough;

    public ThinCrustDough(String dough) {
        this.dough = dough;
    }

    @Override
    public String getDough() {
        return dough;
    }
}
2.2.9、Sauce
package com.erlang.factory.abstract_;

/**
 * @description:
 * @author: erlang
 * @since: 2022-02-09 22:44
 */
public interface Sauce {

    /**
     * 酱汁
     *
     * @return 酱汁
     */
    String getSauce();
}
2.2.10、PlumTomatoSauce
package com.erlang.factory.abstract_;

/**
 * @description:
 * @author: erlang
 * @since: 2022-02-09 22:55
 */
public class PlumTomatoSauce implements Sauce {

    private String sauce;

    public PlumTomatoSauce(String sauce) {
        this.sauce = sauce;
    }

    @Override
    public String getSauce() {
        return sauce;
    }
}
2.2.11、MarinaraSauce
package com.erlang.factory.abstract_;

/**
 * @description:
 * @author: erlang
 * @since: 2022-02-09 22:57
 */
public class MarinaraSauce implements Sauce {

    private String sauce;

    public MarinaraSauce(String sauce) {
        this.sauce = sauce;
    }

    @Override
    public String getSauce() {
        return sauce;
    }
}
2.2.12、CheesePizza
package com.erlang.factory.abstract_;

/**
 * @description: 奶酪披萨
 * @author: erlang
 * @since: 2022-02-08 21:23
 */
public class CheesePizza extends Pizza {

    PizzaIngredientFactory ingredientFactory;

    public CheesePizza(PizzaIngredientFactory ingredientFactory) {
        this.ingredientFactory = ingredientFactory;
    }

    @Override
    public void prepare() {
        System.out.printf("Preparing %s Pizza\n", name);
        dough = ingredientFactory.createDough();
        sauce = ingredientFactory.createSauce();
    }
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值