java设计模式-抽象工厂模式

概念:

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

特点:

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

抽象工厂模式的主要角色 :

  • 抽象工厂(Abstract Factory):提供了创建产品的接口,它包含多个创建产品的方法 new Product(),可以创建多个不同等级的产品。
  • 具体工厂(Concrete Factory):主要是实现抽象工厂中的多个抽象方法,完成具体产品的创建。
  • 抽象产品(Product):定义了产品的规范,描述了产品的主要功能和特性,抽象工厂模式有多个抽象产品。
  • 具体产品(Concrete Product):实现了抽象产品角色所定义的接口,由具体工厂来创建,它同具体工厂之间是多对一的关系
    在这里插入图片描述

代码:

抽象工厂模式的抽象层

package com.example.designmode.bstractfactory;

/**
 * <h3>design-mode</h3>
 * <p>抽象工厂模式的顶级抽象层</p>
 *
 * @author : ZhangYuJie
 * @date : 2022-01-09 20:24
 **/

public interface AbstractFactory {
    /**
     * 定义一个创建图形的方法,让其子类具体实现
     * @param createType
     * @return
     */
    Graph createGraph(String createType);
}

小万绘图的工厂子类

package com.example.designmode.bstractfactory;

/**
 * <h3>design-mode</h3>
 * <p>绘图的工厂子类</p>
 *
 * @author : ZhangYuJie
 * @date : 2022-01-09 20:27
 **/

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 com.example.designmode.bstractfactory;

/**
 * <h3>design-mode</h3>
 * <p>绘图的工厂子类</p>
 *
 * @author : ZhangYuJie
 * @date : 2022-01-09 20:31
 **/

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 com.example.designmode.bstractfactory;

/**
 * <h3>design-mode</h3>
 * <p>图形抽象类</p>
 *
 * @author : ZhangYuJie
 * @date : 2022-01-09 20:24
 **/

public abstract  class Graph {
    /**
     * 开始绘制
     */
    public abstract void startDraw();

    /**
     * 结束绘制
     */
    public abstract void finishDraw();
}

小万绘制圆形的具体实现

package com.example.designmode.bstractfactory;

/**
 * <h3>design-mode</h3>
 * <p>绘制圆形</p>
 *
 * @author : ZhangYuJie
 * @date : 2022-01-09 20:28
 **/

public class WanCircle extends Graph {
    @Override
    public void startDraw() {
        System.out.println("万开始绘制圆形...");
    }

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

小李绘制矩形的具体实现

package com.example.designmode.bstractfactory;

/**
 * <h3>design-mode</h3>
 * <p>绘制矩形</p>
 *
 * @author : ZhangYuJie
 * @date : 2022-01-09 20:28
 **/

public class LiRectangle extends Graph {
    @Override
    public void startDraw() {
        System.out.println("李开始绘制矩形...");
    }

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

绘图的工具类

package com.example.designmode.bstractfactory;

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

/**
 * <h3>design-mode</h3>
 * <p>绘图的工具类</p>
 *
 * @author : ZhangYuJie
 * @date : 2022-01-09 20:29
 **/

public class DrawGraph {

    AbstractFactory factory;

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

    private void setFactory(AbstractFactory factory) {
        Graph graph;
        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("请输入需要绘制的图形形状:");
            return reader.readLine();
        } catch (IOException e) {
            e.printStackTrace();
            return "";
        }
    }
}

图形的具体绘制者

package com.example.designmode.bstractfactory;

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

/**
 * <h3>design-mode</h3>
 * <p>图形绘制者</p>
 *
 * @author : ZhangYuJie
 * @date : 2022-01-09 20:30
 **/

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();
        }
    }
}

演示:

"C:\Program Files\Java\jdk-11.0.11\bin\java.exe" "-javaagent:D:\IntelliJ IDEA 2020.1\lib\idea_rt.jar=56564:D:\IntelliJ IDEA 2020.1\bin" -Dfile.encoding=UTF-8 -classpath D:\workspace2021\design-mode\target\classes;D:\Repository\org\springframework\boot\spring-boot-starter-web\2.6.2\spring-boot-starter-web-2.6.2.jar;D:\Repository\org\springframework\boot\spring-boot-starter\2.6.2\spring-boot-starter-2.6.2.jar;D:\Repository\org\springframework\boot\spring-boot-starter-logging\2.6.2\spring-boot-starter-logging-2.6.2.jar;D:\Repository\ch\qos\logback\logback-classic\1.2.9\logback-classic-1.2.9.jar;D:\Repository\ch\qos\logback\logback-core\1.2.9\logback-core-1.2.9.jar;D:\Repository\org\apache\logging\log4j\log4j-to-slf4j\2.17.0\log4j-to-slf4j-2.17.0.jar;D:\Repository\org\apache\logging\log4j\log4j-api\2.17.0\log4j-api-2.17.0.jar;D:\Repository\org\slf4j\jul-to-slf4j\1.7.32\jul-to-slf4j-1.7.32.jar;D:\Repository\jakarta\annotation\jakarta.annotation-api\1.3.5\jakarta.annotation-api-1.3.5.jar;D:\Repository\org\yaml\snakeyaml\1.29\snakeyaml-1.29.jar;D:\Repository\org\springframework\boot\spring-boot-starter-json\2.6.2\spring-boot-starter-json-2.6.2.jar;D:\Repository\com\fasterxml\jackson\core\jackson-databind\2.13.1\jackson-databind-2.13.1.jar;D:\Repository\com\fasterxml\jackson\core\jackson-annotations\2.13.1\jackson-annotations-2.13.1.jar;D:\Repository\com\fasterxml\jackson\core\jackson-core\2.13.1\jackson-core-2.13.1.jar;D:\Repository\com\fasterxml\jackson\datatype\jackson-datatype-jdk8\2.13.1\jackson-datatype-jdk8-2.13.1.jar;D:\Repository\com\fasterxml\jackson\datatype\jackson-datatype-jsr310\2.13.1\jackson-datatype-jsr310-2.13.1.jar;D:\Repository\com\fasterxml\jackson\module\jackson-module-parameter-names\2.13.1\jackson-module-parameter-names-2.13.1.jar;D:\Repository\org\springframework\boot\spring-boot-starter-tomcat\2.6.2\spring-boot-starter-tomcat-2.6.2.jar;D:\Repository\org\apache\tomcat\embed\tomcat-embed-core\9.0.56\tomcat-embed-core-9.0.56.jar;D:\Repository\org\apache\tomcat\embed\tomcat-embed-el\9.0.56\tomcat-embed-el-9.0.56.jar;D:\Repository\org\apache\tomcat\embed\tomcat-embed-websocket\9.0.56\tomcat-embed-websocket-9.0.56.jar;D:\Repository\org\springframework\spring-web\5.3.14\spring-web-5.3.14.jar;D:\Repository\org\springframework\spring-beans\5.3.14\spring-beans-5.3.14.jar;D:\Repository\org\springframework\spring-webmvc\5.3.14\spring-webmvc-5.3.14.jar;D:\Repository\org\springframework\spring-aop\5.3.14\spring-aop-5.3.14.jar;D:\Repository\org\springframework\spring-context\5.3.14\spring-context-5.3.14.jar;D:\Repository\org\springframework\spring-expression\5.3.14\spring-expression-5.3.14.jar;D:\Repository\org\springframework\boot\spring-boot-devtools\2.6.2\spring-boot-devtools-2.6.2.jar;D:\Repository\org\springframework\boot\spring-boot\2.6.2\spring-boot-2.6.2.jar;D:\Repository\org\springframework\boot\spring-boot-autoconfigure\2.6.2\spring-boot-autoconfigure-2.6.2.jar;D:\Repository\org\projectlombok\lombok\1.18.22\lombok-1.18.22.jar;D:\Repository\org\slf4j\slf4j-api\1.7.32\slf4j-api-1.7.32.jar;D:\Repository\org\springframework\spring-core\5.3.14\spring-core-5.3.14.jar;D:\Repository\org\springframework\spring-jcl\5.3.14\spring-jcl-5.3.14.jar com.example.designmode.bstractfactory.DrawPerson
请输入绘制者的姓名:
xw
请输入需要绘制的图形形状:
circle
万开始绘制圆形...
万结束绘制圆形...
-------------------
请输入需要绘制的图形形状:
rectangle
万开始绘制矩形...
万结束绘制圆形...
-------------------

总结:

抽象工厂模式的扩展有一定的“开闭原则”倾斜性。
当增加一个新的产品族时只需增加一个新的具体工厂,不需要修改原代码,满足开闭原则;
当产品族中需要增加一个新种类的产品时,则所有的工厂类都需要进行修改,不满足开闭原则。
另一方面,当系统中只存在一个等级结构的产品时,抽象工厂模式将退化到工厂方法模式。
从设计层面看,抽象工厂模式就是对简单工厂模式的改进(或者称为进一步的抽象)。
将工厂抽象成两层,Abstract Factory(抽象工厂) 和具体实现的工厂子类。程序员可以根据创建对象类型使用对应的工厂子类。这样将单个的简单工厂类变成了工厂簇, 更利于代码的维护和扩展。
代码地址

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值