23种设计模式——抽象工厂模式

简介: 抽象工厂模式的优点是隔离了具体类的生成,使得客户端不需要知道什么被创建了,而缺点就在于新增新的行为会比较麻烦,因为当添加一个新的产品对象时,需要更改接口及其下的所有子类。
今天我们来学习一下第三种设计模式——抽象工厂模式,
概念:

所谓抽象工厂模式就是提供一个接口,用于创建相关或者依赖对象的家族,而不需要明确指定具体类。它允许客户端使用抽象的接口来创建一组相关的产品,而不需要关心实际产出的具体产品是什么。这样一来,客户就可以从具体的产品中被解耦。它的优点是隔离了具体类的生成,使得客户端不需要知道什么被创建了,而缺点就在于新增新的行为会比较麻烦,因为当添加一个新的产品对象时,需要更改接口及其下的所有子类。

特点:
  • 可以在类的内部对产品族中相关联的多等级产品共同管理,而不必专门引入多个新的类来进行管理。
  • 当增加一个新的产品族时不需要修改原代码,满足开闭原则。
  • 其缺点是:当产品族中需要增加一个新的产品时,所有的工厂类都需要进行修改。
抽象工厂模式的主要角色 :
  • 抽象工厂(Abstract Factory):提供了创建产品的接口,它包含多个创建产品的方法 new Product(),可以创建多个不同等级的产品。
  • 具体工厂(Concrete Factory):主要是实现抽象工厂中的多个抽象方法,完成具体产品的创建。
  • 抽象产品(Product):定义了产品的规范,描述了产品的主要功能和特性,抽象工厂模式有多个抽象产品。
  • 具体产品(Concrete Product):实现了抽象产品角色所定义的接口,由具体工厂来创建,它同具体工厂之间是多对一的关系。
类图分析:

抽象工厂模式类图分析.png

代码:

抽象工厂模式的抽象层

package cn.ppdxzz.abstractfactory;

/**
 * Description:抽象工厂模式的顶级抽象层
 *
 * @Date: 2020/3/5 15:03
 * @Author: PeiChen
 */
public interface AbstractFactory {
    //定义一个创建图形的方法,让其子类具体实现
    Graph createGraph(String createType);
}

小万绘图的工厂子类

package cn.ppdxzz.abstractfactory;

/**
 * Description:小万绘图的工厂子类
 *
 * @Date: 2020/3/5 15:18
 * @Author: PeiChen
 */
public class WanFactory implements AbstractFactory{

    @Override
    public Graph createGraph(String createType) {
        Graph graph = null;
        if ("circle".equals(createType)) {
            graph = new WanCircle();
        }else if ("rectangle".equals(createType)) {
            graph = new WanRectangle();
        }
        return graph;
    }
}

小李绘图的工厂子类

package cn.ppdxzz.abstractfactory;

/**
 * Description:小李绘图的工厂子类
 *
 * @Date: 2020/3/5 15:19
 * @Author: PeiChen
 */
public class LiFactory implements AbstractFactory {
    @Override
    public Graph createGraph(String createType) {
        Graph graph = null;
        if ("circle".equals(createType)) {
            graph = new LiCircle();
        }else if ("rectangle".equals(createType)) {
            graph = new LiRectangle();
        }
        return graph;
    }
}

图形抽象类

package cn.ppdxzz.abstractfactory;

/**
 * Description:图形抽象类
 *
 * @Date: 2020/3/5 15:06
 * @Author: PeiChen
 */
public abstract class Graph {
    //开始绘制
    public abstract void startDraw();
    //结束绘制
    public abstract void finishDraw();
}

小万绘制圆形的具体实现

package cn.ppdxzz.abstractfactory;

/**
 * Description:小万绘制圆形
 *
 * @Date: 2020/3/5 15:06
 * @Author: PeiChen
 */
public class WanCircle extends Graph {
    @Override
    public void startDraw() {
        System.out.println("小万开始绘制圆形...");
    }

    @Override
    public void finishDraw() {
        System.out.println("小万结束绘制圆形...");
        System.out.println("-------------------");
    }
    
}

小李绘制矩形的具体实现

package cn.ppdxzz.abstractfactory;

/**
 * Description:小李绘制矩形
 *
 * @Date: 2020/3/5 15:06
 * @Author: PeiChen
 */
public class LiRectangle extends Graph {
    @Override
    public void startDraw() {
        System.out.println("小李开始绘制矩形...");
    }

    @Override
    public void finishDraw() {
        System.out.println("小李结束绘制矩形...");
        System.out.println("-------------------");
    }
}

绘图的工具类

package cn.ppdxzz.abstractfactory;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;

/**
 * Description:绘图的工具类
 *
 * @Date: 2020/3/5 15:31
 * @Author: PeiChen
 */
public class DrawGraph {

    AbstractFactory factory;

    //构造方法
    public DrawGraph(AbstractFactory factory) {
        setFactory(factory);
    }

    private void setFactory(AbstractFactory factory) {
        Graph graph = null;
        String createType = "";
        this.factory = factory;
        do {
            createType = getType();
            //factory可能是小万的工厂子类,也可能是小李的工厂子类
            graph = factory.createGraph(createType);
            if (graph != null) {
                graph.startDraw();
                graph.finishDraw();
            }else {
                System.out.println("输入有误,您已退出!");
                break;
            }

        }while (true);

    }

    //获取绘制者绘制的图形形状
    private String getType() {
        try {
            BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
            System.out.println("请输入需要绘制的图形形状:");
            String str = reader.readLine();
            return str;
        } catch (IOException e) {
            e.printStackTrace();
            return "";
        }
    }
}

图形的具体绘制者

package cn.ppdxzz.abstractfactory;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;

/**
 * Description:图形绘制者
 *
 * @Date: 2020/3/5 15:57
 * @Author: PeiChen
 */
public class DrawPerson {
    public static void main(String[] args) {
        System.out.println("请输入绘制者的姓名:");
        try {
            BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
            String name = reader.readLine();
            if ("xw".equals(name)) {
                new DrawGraph(new WanFactory());
            }else {
                new DrawGraph(new LiFactory());
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

演示:

抽象方法演示.png

JDK源码分析(Calendar类就属于简单工厂模式,这个我没有讲解,因为他不属于23种设计模式):

JDK_Calender简单工厂模式.png

总结:
  • 抽象工厂模式的扩展有一定的“开闭原则”倾斜性。

    • 当增加一个新的产品族时只需增加一个新的具体工厂,不需要修改原代码,满足开闭原则;
    • 当产品族中需要增加一个新种类的产品时,则所有的工厂类都需要进行修改,不满足开闭原则。
  • 另一方面,当系统中只存在一个等级结构的产品时,抽象工厂模式将退化到工厂方法模式。
  • 从设计层面看,抽象工厂模式就是对简单工厂模式的改进(或者称为进一步的抽象)。
  • 将工厂抽象成两层,Abstract Factory(抽象工厂) 和具体实现的工厂子类。程序员可以根据创建对象类型使用对应的工厂子类。这样将单个的简单工厂类变成了工厂簇, 更利于代码的维护和扩展。
抽象工厂模式就讲解到这里吧,下一个设计模式是建造者模式,敬请期待。
目录
相关文章
|
4月前
|
设计模式 PHP
php设计模式--抽象工厂模式(二)
php设计模式--抽象工厂模式(二)
26 0
|
3月前
|
设计模式
**工厂模式与抽象工厂模式**都是创建型设计模式,用于封装对象创建,减少耦合
【6月更文挑战第23天】**工厂模式与抽象工厂模式**都是创建型设计模式,用于封装对象创建,减少耦合。工厂模式专注于单个对象,通过具体工厂创建具体产品,适用于简单对象创建;抽象工厂则关注一系列相关产品,提供创建一族对象的接口,适用于处理多个不兼容产品族。选择模式基于问题域的复杂性,单个产品需求时用工厂模式,多产品族时用抽象工厂模式。
28 5
|
4月前
|
设计模式 Java
【设计模式】JAVA Design Patterns——Abstract Factory(抽象工厂模式)
【设计模式】JAVA Design Patterns——Abstract Factory(抽象工厂模式)
|
4月前
|
设计模式 Java
Java一分钟之-设计模式:工厂模式与抽象工厂模式
【5月更文挑战第17天】本文探讨了软件工程中的两种创建型设计模式——工厂模式和抽象工厂模式。工厂模式提供了一个创建对象的接口,延迟实例化到子类决定。过度使用或违反单一职责原则可能导致问题。代码示例展示了如何创建形状的工厂。抽象工厂模式则用于创建一系列相关对象,而不指定具体类,但添加新产品可能需修改现有工厂。代码示例展示了创建颜色和形状的工厂。根据需求选择模式,注意灵活性和耦合度。理解并恰当运用这些模式能提升代码质量。
52 2
|
13天前
|
设计模式 Java
Java 设计模式之谜:工厂模式与抽象工厂模式究竟隐藏着怎样的神奇力量?
【8月更文挑战第30天】在Java编程中,设计模式为常见问题提供了高效解决方案。工厂模式与抽象工厂模式是常用的对象创建型设计模式,能显著提升代码的灵活性、可维护性和可扩展性。工厂模式通过定义创建对象的接口让子类决定实例化哪个类;而抽象工厂模式则进一步提供了一个创建一系列相关或相互依赖对象的接口,无需指定具体类。这种方式使得系统更易于扩展和维护。
26 1
|
4月前
|
设计模式 Java
【设计模式系列笔记】抽象工厂模式
抽象工厂模式(Abstract Factory Pattern)是一种设计模式,属于创建型模式之一。它提供了一种方式来创建一系列相关或相互依赖的对象,而无需指定它们具体的类。抽象工厂模式通过引入抽象的工厂接口,使得客户端代码可以使用抽象的接口来创建一组相关的产品,而不关心这些产品的具体实现。
155 4
|
24天前
|
设计模式 XML 存储
【三】设计模式~~~创建型模式~~~抽象工厂模式(Java)
文章详细介绍了抽象工厂模式,这是一种创建型设计模式,用于提供一个接口以创建一系列相关或相互依赖的对象,而不指定它们具体的类。通过代码示例和结构图,文章展示了抽象工厂模式的动机、定义、结构、优点、缺点以及适用场景,并探讨了如何通过配置文件和反射机制实现工厂的动态创建。
【三】设计模式~~~创建型模式~~~抽象工厂模式(Java)
|
29天前
|
设计模式 Java C语言
设计模式-----------工厂模式之抽象工厂模式(创建型)
抽象工厂模式是一种创建型设计模式,它提供了一个接口用于创建一系列相关或相互依赖的对象,而无需指定具体类,从而增强了程序的可扩展性并确保客户端只使用同一产品族的产品。
设计模式-----------工厂模式之抽象工厂模式(创建型)
|
1月前
|
设计模式 存储 XML
[设计模式]创建型模式-抽象工厂模式
[设计模式]创建型模式-抽象工厂模式
|
3月前
|
设计模式 SQL 关系型数据库
抽象工厂模式-大话设计模式
抽象工厂模式-大话设计模式