• 【创建型模式】抽象工厂模式


    一、抽象工厂模式概述

            抽象工厂模式定义提供一个创建一系列相关或相互依赖对象的接口,而无须指定它们具体的类。

    • 模式动机
      • 1.当系统提供的工厂生产的具体产品并不是一个简单的对象,而是多个位于不同产品等级结构、属于不同类型的具体产品时就可以使用抽象工厂模式;
      • 2.抽象工厂模式是所有形式的工厂模式中最为抽象和最具一般性的一种形式。
    • 抽象工厂:一个工厂可以生产一系列(一族产品),极大减少了工厂类的数量。又叫做工具模式:抽象工厂模式中的具体工厂不只是创建一种产品,她负责一族产品;当一个工厂等级结构可以创建出分属于不同产品登记结构的一个产品族中的所有对象时,抽象工厂模式比工厂方法模式更为简单、更有效率。

            产品族是指位于不同等级结构中,功能相关联的产品组成的家族。

            产品等级:产品的继承结构。

    • 开闭原则的倾斜性
      • 1.增加产品族:抽象工厂模式很好地支持了开闭原则,只需要增加具体产品并对应增加了一个新的具体工厂,对已有的代码无须做任何修改。
      • 2.增加新的产品等级结构:需要修改所有的工厂角色,包括抽象工厂类,在所有的工厂类中都需要增加生产新产品的方法,违背了开闭原则。
    • 抽象工厂模式的优缺点
      • 优点
        • 1.隔离了具体类的生成,使得客户端并不需要知道什么被创建;
        • 2.当一个产品族中的多个对象被设计成一起工作时,它能够保证客户端始终只使用同一个产品族中的对象;
        • 3.增加新的产品族很方便,无需修改已有系统,符合开闭原则。
      • 缺点
        • 增加心得产品等级结构麻烦,需要对原有系统进行较大的修改,甚至需要修改抽象层代码,这显然会带来较大的不便。违背了开闭原则。
    • 模式适用环境
      • 1.当一个系统不应当依赖于产品类实例如何被创建、组合和表达的细节;
      • 2.系统中有多于一个的产品族,但每次只使用其中一个产品族;
      • 3.属于同一个产品族的产品将在一起使用,这一约束必须在系统的设计中体现出来;
      • 4.产品等级结构稳定,在设计完成之后不会向系统中增加新的产等级结构或者删除已有的产品等级结构。

    二、代码实现

            结构包含四个角色:

    • 抽象工厂(Abstract Factory):一个工厂可以生产一系列(一族产品),极大减少了工厂类的数量。
    • 具体工厂(ConcreteFactory):实现了在抽象工厂中声明的创建产品的方法,生成了一组具体产品,这些产品构成了一个产品族,每个产品都位于某个产品等级结构中。
    • 抽象产品(AbstractProduct):它为每种产品声明接口,在抽象产品中声明了产品的所有业务方法。
    • 具体产品(Concrete Product​​​​​​​):定义具体工厂生产的具体产品对象,实现在抽象产品接口中声明的业务方法。
            2.1 抽象工厂(两个抽象工厂:原料工厂和披萨总店)
    1. package abstractFactory.Pizza;
    2. //抽象工厂:披萨原料工厂
    3. public interface PizzaIngredientFactory {
    4. //封装了制造产品原料的方法,其中每个原料都是一个类
    5. public Dough createDough();//面团
    6. public Sauce createSauce();//酱料
    7. public Cheese createCheese();//芝士
    8. public Clams createClam();
    9. }
    1. package abstractFactory.Pizza;
    2. //抽象工厂:抽象披萨店(总店)
    3. public abstract class PizzaStore {
    4. //生产具体类型的披萨店(每一个子店口味不一样,这里就是产生不同子店)
    5. protected abstract Pizza createPizza(String item);
    6. public Pizza orderPizza(String type) {
    7. Pizza pizza = createPizza(type);
    8. System.out.println("--- Making a " + pizza.getName() + " ---");
    9. pizza.prepare();
    10. pizza.bake();
    11. pizza.cut();
    12. pizza.box();
    13. return pizza;
    14. }
    15. }
            2.2 具体工厂(芝加哥原料厂、纽约原料厂和芝加哥分店、纽约分店)
    1. package abstractFactory.Pizza;
    2. //芝加哥原料工厂
    3. public class ChicagoPizzaIngredientFactory implements PizzaIngredientFactory {
    4. @Override
    5. public Dough createDough() {
    6. // TODO Auto-generated method stub
    7. return new ThickCrustDough();
    8. }
    9. @Override
    10. public Sauce createSauce() {
    11. // TODO Auto-generated method stub
    12. return new PlumTomatoSauce();
    13. }
    14. @Override
    15. public Cheese createCheese() {
    16. // TODO Auto-generated method stub
    17. return new MozzarellaCheese();
    18. }
    19. @Override
    20. public Clams createClam() {
    21. // TODO Auto-generated method stub
    22. return new FrozenClams();
    23. }
    24. }
    1. package abstractFactory.Pizza;
    2. //纽约原料工厂
    3. public class NYPizzaIngredientFactory implements PizzaIngredientFactory {
    4. @Override
    5. public Dough createDough() {
    6. // TODO Auto-generated method stub
    7. return new ThinCrustDough();
    8. }
    9. @Override
    10. public Sauce createSauce() {
    11. // TODO Auto-generated method stub
    12. return new MarinaraSauce();
    13. }
    14. @Override
    15. public Cheese createCheese() {
    16. // TODO Auto-generated method stub
    17. return new ReggianoCheese();
    18. }
    19. @Override
    20. public Clams createClam() {
    21. // TODO Auto-generated method stub
    22. return new FreshClams();
    23. }
    24. }
    1. package abstractFactory.Pizza;
    2. //具体披萨分店:芝加哥分店
    3. public class ChicagoPizzaStore extends PizzaStore {
    4. @Override
    5. protected Pizza createPizza(String item) {
    6. // TODO Auto-generated method stub
    7. Pizza pizza = null;
    8. PizzaIngredientFactory ingredientFactory =
    9. new ChicagoPizzaIngredientFactory();
    10. if (item.equals("cheese")) {
    11. pizza = new CheesePizza(ingredientFactory);
    12. pizza.setName("Chicago Style Cheese Pizza");
    13. } else if (item.equals("clam")) {
    14. pizza = new ClamPizza(ingredientFactory);
    15. pizza.setName("Chicago Style Clam Pizza");
    16. }
    17. return pizza;
    18. }
    19. }
    1. package abstractFactory.Pizza;
    2. //具体披萨分店:纽约分店
    3. public class NYPizzaStore extends PizzaStore {
    4. @Override
    5. protected Pizza createPizza(String item) {
    6. // TODO Auto-generated method stub
    7. Pizza pizza = null;
    8. PizzaIngredientFactory ingredientFactory = new NYPizzaIngredientFactory();
    9. if (item.equals("cheese")) {
    10. pizza = new CheesePizza(ingredientFactory);
    11. pizza.setName("New York Style Cheese Pizza");
    12. } else if (item.equals("clam")) {
    13. pizza = new ClamPizza(ingredientFactory);
    14. pizza.setName("New York Style Clam Pizza");
    15. }
    16. return pizza;
    17. }
    18. }
            2.3 抽象产品(原料抽象产品:Dough、Sauce、Cheese、Clam;披萨抽象产品:Pizza) 
    1. package abstractFactory.Pizza;
    2. //抽象产品:披萨原料接口,面团
    3. public interface Dough {
    4. public String toString();
    5. }
    1. package abstractFactory.Pizza;
    2. //抽象产品:披萨原料接口,酱料
    3. public interface Sauce {
    4. public String toString();
    5. }
    1. package abstractFactory.Pizza;
    2. //抽象产品:披萨原料接口,芝士
    3. public interface Cheese {
    4. public String toString();
    5. }
    1. package abstractFactory.Pizza;
    2. //抽象产品:,披萨原料接口
    3. public interface Clams {
    4. public String toString();
    5. }
    1. package abstractFactory.Pizza;
    2. //抽象产品:披萨抽象类
    3. public abstract class Pizza {
    4. String name;
    5. Dough dough;
    6. Sauce sauce;
    7. Cheese cheese;
    8. Clams clam;
    9. abstract void prepare();
    10. void bake() {
    11. System.out.println("Bake for 25 minutes at 350");
    12. }
    13. void cut() {
    14. System.out.println("Cutting the pizza into diagonal slices");
    15. }
    16. void box() {
    17. System.out.println("Place pizza in official PizzaStore box");
    18. }
    19. void setName(String name) {
    20. this.name = name;
    21. }
    22. String getName() {
    23. return name;
    24. }
    25. public String toString() {
    26. StringBuffer result = new StringBuffer();
    27. result.append("---- " + name + " ----\n");
    28. if (dough != null) {
    29. result.append(dough);
    30. result.append("\n");
    31. }
    32. if (sauce != null) {
    33. result.append(sauce);
    34. result.append("\n");
    35. }
    36. if (cheese != null) {
    37. result.append(cheese);
    38. result.append("\n");
    39. }
    40. if (clam != null) {
    41. result.append(clam);
    42. result.append("\n");
    43. }
    44. return result.toString();
    45. }
    46. }
            2.4 具体产品(具体原料:ThickCrustDough、ThinCrustDough、PlumTomatoSauce、MarinaraSauce、MozzarellaCheese、ReggianoCheese、FrozenClams、FreshClams;具体披萨:CheesePizza、ClamPizza)
    1. package abstractFactory.Pizza;
    2. //具体披萨原料,对应的原料工厂:ChicagoPizzaIngredientFactory
    3. public class ThickCrustDough implements Dough {
    4. public String toString() {
    5. return "ThickCrust style extra thick crust dough";
    6. }
    7. }
    1. package abstractFactory.Pizza;
    2. //具体披萨原料,对应的原料工厂:NYPizzaIngredientFactory
    3. public class ThinCrustDough implements Dough {
    4. public String toString() {
    5. return "Thin Crust Dough";
    6. }
    7. }
    1. package abstractFactory.Pizza;
    2. //具体披萨原料,对应原料工厂:ChicagoPizzaIngredientFactory
    3. public class PlumTomatoSauce implements Sauce {
    4. public String toString() {
    5. return "Tomato sauce with plum tomatoes";
    6. }
    7. }
    1. package abstractFactory.Pizza;
    2. //具体披萨原料,对应的原料工厂:NYPizzaIngredientFactory
    3. public class MarinaraSauce implements Sauce {
    4. public String toString() {
    5. return "Marinara Sauce";
    6. }
    7. }
    1. package abstractFactory.Pizza;
    2. //具体披萨原料,对应原料工厂ChicagoPizzaIngredientFactory
    3. public class MozzarellaCheese implements Cheese {
    4. public String toString() {
    5. return "Shredded Mozzarella";
    6. }
    7. }
    1. package abstractFactory.Pizza;
    2. //具体披萨原料,对应原料工厂NYPizzaIngredientFactory
    3. public class ReggianoCheese implements Cheese {
    4. public String toString() {
    5. return "Reggiano Cheese";
    6. }
    7. }
    1. package abstractFactory.Pizza;
    2. //具体披萨原料,对应的原料工厂:ChicagoPizzaIngredientFactory
    3. public class FrozenClams implements Clams {
    4. public String toString() {
    5. return "Frozen Clams from Chesapeake Bay";
    6. }
    7. }
    1. package abstractFactory.Pizza;
    2. //具体披萨原料,对应的原料工厂:NYPizzaIngredientFactory
    3. public class FreshClams implements Clams {
    4. public String toString() {
    5. return "Fresh Clams from Long Island Sound";
    6. }
    7. }
    1. package abstractFactory.Pizza;
    2. //具体披萨子类
    3. public class CheesePizza extends Pizza {
    4. PizzaIngredientFactory ingredientFactory;
    5. public CheesePizza(PizzaIngredientFactory ingredientFactory) {
    6. this.ingredientFactory = ingredientFactory;
    7. }
    8. @Override
    9. void prepare() {
    10. // TODO Auto-generated method stub
    11. System.out.println("Preparing " + name);
    12. dough = ingredientFactory.createDough();
    13. sauce = ingredientFactory.createSauce();
    14. cheese = ingredientFactory.createCheese();
    15. }
    16. }
    1. package abstractFactory.Pizza;
    2. //具体披萨子类
    3. public class ClamPizza extends Pizza {
    4. PizzaIngredientFactory ingredientFactory;
    5. public ClamPizza(PizzaIngredientFactory ingredientFactory) {
    6. this.ingredientFactory = ingredientFactory;
    7. }
    8. @Override
    9. void prepare() {
    10. // TODO Auto-generated method stub
    11. System.out.println("Preparing " + name);
    12. dough = ingredientFactory.createDough();
    13. sauce = ingredientFactory.createSauce();
    14. cheese = ingredientFactory.createCheese();
    15. }
    16. }
            2.5 main方法实现抽象工厂模式
    1. package abstractFactory.Pizza;
    2. public class Test {
    3. public static void main(String[] args) {
    4. // TODO Auto-generated method stub
    5. PizzaStore nyStore = new NYPizzaStore();
    6. PizzaStore chicagoStore = new ChicagoPizzaStore();
    7. //纽约子店点餐
    8. Pizza pizza = nyStore.orderPizza("cheese");
    9. System.out.println("Ethan ordered a " + pizza + "\n");
    10. //芝加哥子店点餐
    11. pizza = chicagoStore.orderPizza("cheese");
    12. System.out.println("Joel ordered a " + pizza + "\n");
    13. //纽约子店点餐
    14. pizza = nyStore.orderPizza("clam");
    15. System.out.println("Ethan ordered a " + pizza + "\n");
    16. //芝加哥子店点餐
    17. pizza = chicagoStore.orderPizza("clam");
    18. System.out.println("Joel ordered a " + pizza + "\n");
    19. }
    20. }
            2.6 UML图

    三、代码结构图

    四、工厂方法模式与抽象工厂模式的总结

    • 所有的工厂都是用来封装对象的创建
    • 简单工厂:虽然不是真正的设计模式,但仍不失为一个简单的方法,可以将客户程序从具体类解耦;
    • 工厂方法使用继承:把对象的创建委托给子类,子类实现工厂方法来创建对象;
    • 抽象工厂使用对象组合:对象的创建被实现在工厂接口所暴露出来的方法中;
    • 所有工厂模式都通过减少应用程序和具体类之间的依赖促进松耦合;
    • 工厂方法允许类将实例化延迟到子类进行。
    • 抽象工厂创建相关的对象家族,而不需要依赖它们的具体类。

            UML图对比

            工厂是很有威力的技巧,帮助我们针对抽象编程,而不要针对具体类编程​​​​​​​

  • 相关阅读:
    记一次内网靶机实战
    Linux服务器使用Nginx和Tomcat配置Https请求
    uniapp使用Canvas实现电子签名
    嚼一嚼Halcon中的3D手眼标定
    CSS特殊学习网址
    oCPC实践录 | oCPC转化的设计、选择、归因与成本设置(3)
    jQuery 中的 bind(), live(), delegate(), on()的区别
    前端加密数据 后端java对应解密数据的方案
    金仓数据库KingbaseES客户端编程接口指南-ado.net(2. 概述)
    pr视频剪辑素材,免费下载
  • 原文地址:https://blog.csdn.net/qq_45276194/article/details/137916018