【从零开始学设计模式】第四章_抽象工厂模式(与工厂模式区分)

第四章_抽象工厂模式(与工厂模式区分)

1.介绍

1.1定义

为访问类提供一个创建一组相关或相互依赖对象的接口,且==访问类无须指定所要产品的具体类==就能得到同族的不同等级的产品的模式结构;

1.2解决的问题

主要解决接口选择的问题。

1.3应用实例

  • 系统中有多个产品族, 每个具体工厂创建同族但属于不同等级的产品;
  • 系统一次只可能消费其中某一族产品,即同族的产品一起使用。

1、QQ 换皮肤,一整套一起换。 2、生成不同操作系统的程序。

1.5角色

  • 抽象产品(Abstract Product):定义了一组产品对象的共同接口或抽象类,描述了产品对象的公共方法。

  • 具体产品(Concrete Product):实现了抽象产品接口,定义了具体产品的特定行为和属性。

  • 抽象工厂(Abstract Factory):声明了一组用于创建产品对象的方法,每个方法对应一种产品类型。抽象工厂可以是接口或抽象类。

  • 具体工厂(Concrete Factory):实现了抽象工厂接口,负责创建具体产品对象的实例。

2.举例

创建 Shape 接口和实现这些接口的实体类。下一步是创建抽象工厂类 AbstractFactory 。接着定义工厂类 ShapeFactoryColorFactory ,这两个工厂类都是实现了 AbstractFactory 。然后创建一个工厂创造器/生成器类 FactoryProducer

AbstractFactoryPatternDemo 类使用 FactoryProducer 来获取 AbstractFactory 对象。它将向 AbstractFactory 传递形状信息 ShapeCIRCLE / RECTANGLE / SQUARE ),以便获取它所需对象的类型。同时它还向 AbstractFactory 传递颜色信息 ColorRED / GREEN / BLUE),以便获取它所需对象的类型。

  • 步骤1:为形状和颜色创建一个接口
java 复制代码
public interface Shape {
   void draw();
}
java 复制代码
public interface Color {
   void fill();
}
  • 步骤2:分别创建实现形状和颜色的实体类
java 复制代码
//实现形状接口的实体类
public class Rectangle implements Shape {
 
   @Override
   public void draw() {
      System.out.println("Inside Rectangle::draw() method.");
   }
}

public class Square implements Shape {
 
   @Override
   public void draw() {
      System.out.println("Inside Square::draw() method.");
   }
}

public class Circle implements Shape {
 
   @Override
   public void draw() {
      System.out.println("Inside Circle::draw() method.");
   }
}
java 复制代码
//实现颜色接口的实体类
public class Red implements Color {
 
   @Override
   public void fill() {
      System.out.println("Inside Red::fill() method.");
   }
}

public class Green implements Color {
 
   @Override
   public void fill() {
      System.out.println("Inside Green::fill() method.");
   }
}

public class Blue implements Color {
 
   @Override
   public void fill() {
      System.out.println("Inside Blue::fill() method.");
   }
}
  • 步骤3:为Color和Shape接口对象创建抽象类来获取工厂
java 复制代码
public abstract class AbstractFactory {
   public abstract Color getColor(String color);
   public abstract Shape getShape(String shape);
}
  • 步骤4:创建继承AbstractFactory的工厂类,基于给定的信息生成实体类的对象
java 复制代码
//ShapeFactory:Shape工厂
public class ShapeFactory extends AbstractFactory {
    
   @Override
   public Shape getShape(String shapeType){
      if(shapeType == null){
         return null;
      }        
      if(shapeType.equalsIgnoreCase("CIRCLE")){
         return new Circle();
      } else if(shapeType.equalsIgnoreCase("RECTANGLE")){
         return new Rectangle();
      } else if(shapeType.equalsIgnoreCase("SQUARE")){
         return new Square();
      }
      return null;
   }
   
   @Override
   public Color getColor(String color) {
      return null;
   }
}
java 复制代码
//ShapeFactory:Shape工厂
public class ColorFactory extends AbstractFactory {
    
   @Override
   public Shape getShape(String shapeType){
      return null;
   }
   
   @Override
   public Color getColor(String color) {
      if(color == null){
         return null;
      }        
      if(color.equalsIgnoreCase("RED")){
         return new Red();
      } else if(color.equalsIgnoreCase("GREEN")){
         return new Green();
      } else if(color.equalsIgnoreCase("BLUE")){
         return new Blue();
      }
      return null;
   }
}
  • 步骤5:创建一个工厂创造器,通过传递形状或颜色信息来获取指定的工厂
java 复制代码
public class FactoryProducer {
   public static AbstractFactory getFactory(String choice){
      if(choice.equalsIgnoreCase("SHAPE")){
         return new ShapeFactory();
      } else if(choice.equalsIgnoreCase("COLOR")){
         return new ColorFactory();
      }
      return null;
   }
}
  • 消费者:通过FactoryProducer来获取AbstracFactory,进而通过传递类型信息来获取实体类对象
java 复制代码
public class AbstractFactoryPatternDemo {
   public static void main(String[] args) {
 
      //获取形状工厂
      AbstractFactory shapeFactory = FactoryProducer.getFactory("SHAPE");
 
      //获取形状为 Circle 的对象
      Shape shape1 = shapeFactory.getShape("CIRCLE");
 
      //调用 Circle 的 draw 方法
      shape1.draw();
 
      //获取形状为 Rectangle 的对象
      Shape shape2 = shapeFactory.getShape("RECTANGLE");
 
      //调用 Rectangle 的 draw 方法
      shape2.draw();
      
      //获取形状为 Square 的对象
      Shape shape3 = shapeFactory.getShape("SQUARE");
 
      //调用 Square 的 draw 方法
      shape3.draw();
 
      //获取颜色工厂
      AbstractFactory colorFactory = FactoryProducer.getFactory("COLOR");
 
      //获取颜色为 Red 的对象
      Color color1 = colorFactory.getColor("RED");
 
      //调用 Red 的 fill 方法
      color1.fill();
 
      //获取颜色为 Green 的对象
      Color color2 = colorFactory.getColor("GREEN");
 
      //调用 Green 的 fill 方法
      color2.fill();
 
      //获取颜色为 Blue 的对象
      Color color3 = colorFactory.getColor("BLUE");
 
      //调用 Blue 的 fill 方法
      color3.fill();
   }
}
  • 执行结果

3.优缺点

3.1优点

  • 可以在类的内部对产品族中相关联的多等级产品共同管理,而不必专门引入多个新的类进行管理;
  • 当增加一个新的产品族时不需要修改原代码,满足开闭原则。

3.2缺点

  • 当产品族中需要增加一个新的产品时,所有的工厂类都需要进行修改。

4.小结

  • 工厂模式和抽象工厂的异同
    • 工厂模式和抽象工厂都是创建对象的设计模式
    • 创建对象的方式和对象的组织结构不同
      • 工厂模式式通过一个工厂类来创建一个具体的对象
      • 抽象工厂是通过一组相关的工厂来创建一组相关的对象
      • 工厂模式只适用于单一产品的创建
      • 抽象工厂适用于一组相关产品的创建

祥讲创建对象的过程:

  • 工厂模式:定义了一个用于创建对象的接口(工厂),让其具体的工厂子类决定实例化哪个类:使得一个类的实例化延迟到其子类;适用于创建的对象比较少的情况

  • 抽象工厂:定义了一个工厂创作器(用于创建指定的工厂,里面包含多个工厂),进而通过该工厂创建具体的产品

访问类提供一个创建一组相关或相互依赖对象的接口,且==访问类无须指定所要产品的具体类==就能得到同族的不同等级的产品的模式结构;

相关推荐
F-2H12 分钟前
C语言:指针4(常量指针和指针常量及动态内存分配)
java·linux·c语言·开发语言·前端·c++
苹果酱056715 分钟前
「Mysql优化大师一」mysql服务性能剖析工具
java·vue.js·spring boot·mysql·课程设计
_oP_i1 小时前
Pinpoint 是一个开源的分布式追踪系统
java·分布式·开源
mmsx1 小时前
android sqlite 数据库简单封装示例(java)
android·java·数据库
武子康2 小时前
大数据-258 离线数仓 - Griffin架构 配置安装 Livy 架构设计 解压配置 Hadoop Hive
java·大数据·数据仓库·hive·hadoop·架构
豪宇刘3 小时前
MyBatis的面试题以及详细解答二
java·servlet·tomcat
秋恬意3 小时前
Mybatis能执行一对一、一对多的关联查询吗?都有哪些实现方式,以及它们之间的区别
java·数据库·mybatis
FF在路上4 小时前
Knife4j调试实体类传参扁平化模式修改:default-flat-param-object: true
java·开发语言
真的很上进4 小时前
如何借助 Babel+TS+ESLint 构建现代 JS 工程环境?
java·前端·javascript·css·react.js·vue·html
众拾达人4 小时前
Android自动化测试实战 Java篇 主流工具 框架 脚本
android·java·开发语言