Java 设计模式系列:简单工厂模式

简介

简单工厂模式(Simple Factory Pattern)是一种常用的设计模式,属于创建型模式。又叫做静态工厂方法(Static Factory Method)模式,但不属于23种GOF设计模式之一。它提供了一种创建对象的最佳方式。在简单工厂模式中,有一个专门的工厂类负责创建其他类的实例,而客户端无需知道所创建对象的详细信息。工厂类负责实现创建所有实例的内部逻辑,可以创建具有共同父类的不同类的新实例。

在实现上,简单工厂模式的UML类图包含一个工厂类(Creator)、一个抽象产品(Product)和一个具体产品(Concrete Product)。工厂类根据传入的参数动态决定应该创建哪一个产品类的实例。抽象产品是所有创建对象的父类,它负责描述所有实例所共有的公共接口。具体产品是抽象产品的具体实现,是简单工厂模式的创建目标。

简单工厂模式的适用场景包括工厂类负责创建的对象比较少,客户端只需要知道传入工厂类的参数,对于如何创建对象(逻辑)不关心。这种模式有助于减少客户端代码的复杂度,使得客户端无需了解对象的创建细节。

简单工厂模式的优点包括:客户端只需要传入一个正确的参数,就可以获取所需的对象,而无需关心对象的创建细节;工厂类可以封装对象的创建逻辑,客户端无需了解对象的具体创建过程。然而,简单工厂模式的缺点也很明显:工厂类的职责相对过重,增加新的产品需要修改工厂类的判断逻辑,违背了开闭原则;如果工厂类异常,可能会导致整个系统的异常。

示例

首先,定义一个抽象产品类 Product,以及两个具体产品类 ConcreteProductAConcreteProductB,它们都实现了 Product 接口:

java 复制代码
// 抽象产品类  
public interface Product {  
    void use();  
}  
  
// 具体产品类A  
public class ConcreteProductA implements Product {  
    @Override  
    public void use() {  
        System.out.println("使用具体产品A");  
    }  
}  
  
// 具体产品类B  
public class ConcreteProductB implements Product {  
    @Override  
    public void use() {  
        System.out.println("使用具体产品B");  
    }  
}

接下来,定义工厂类 SimpleFactory,它负责根据客户端的需求创建具体产品对象:

java 复制代码
// 工厂类  
public class SimpleFactory {  
    // 使用静态方法,不需要创建工厂实例  
    public static Product createProduct(String type) {  
        if ("A".equalsIgnoreCase(type)) {  
            return new ConcreteProductA();  
        } else if ("B".equalsIgnoreCase(type)) {  
            return new ConcreteProductB();  
        } else {  
            return null;  
        }  
    }  
}

最后,客户端代码使用工厂类来获取产品对象,并调用其方法:

java 复制代码
public class Client {  
    public static void main(String[] args) {  
        // 客户端只需要知道传入工厂类的参数,无需了解创建细节  
        try{
           do{
            BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
            System.out.println("input product type:");
            String type = reader.readLine();
            Product product = SimpleFactory.createProduct(type);   
            if(product!=null){
               // 使用产品  
               product.use();  
            }
             else {
                break;
             }
           }while(true);
        }
        catch (IOException e) {
            e.printStackTrace();
        }
    }  
}

运行客户端代码后,输出将会是:

复制代码
使用具体产品A  
使用具体产品B
相关推荐
Otaku love travel1 小时前
老系统改造增加初始化,自动化数据源配置(tomcat+jsp+springmvc)
java·tomcat·初始化·动态数据源
DKPT1 小时前
Java设计模式之行为型模式(责任链模式)介绍与说明
java·笔记·学习·观察者模式·设计模式
L_autinue_Star1 小时前
手写vector容器:C++模板实战指南(从0到1掌握泛型编程)
java·c语言·开发语言·c++·学习·stl
晨岳1 小时前
CentOS 安装 JDK+ NGINX+ Tomcat + Redis + MySQL搭建项目环境
java·redis·mysql·nginx·centos·tomcat
执笔诉情殇〆2 小时前
前后端分离(java) 和 Nginx在服务器上的完整部署方案(redis、minio)
java·服务器·redis·nginx·minio
YuTaoShao2 小时前
【LeetCode 热题 100】24. 两两交换链表中的节点——(解法一)迭代+哨兵
java·算法·leetcode·链表
程序员的世界你不懂2 小时前
(20)Java+Playwright自动化测试- 操作鼠标拖拽 - 上篇
java·python·计算机外设
AI360labs_atyun2 小时前
Java在AI时代的演进与应用:一个务实的视角
java·开发语言·人工智能·科技·学习·ai
不像程序员的程序媛3 小时前
redis的一些疑问
java·redis·mybatis
知其然亦知其所以然3 小时前
Java 面试高频题:GC 到底回收了什么、怎么回收、啥时候回收?
java·后端·面试