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
相关推荐
澈2073 分钟前
C++并查集:高效解决连通性问题
java·c++·算法
2401_873479402 小时前
运营活动被薅羊毛怎么防?用IP查询+设备指纹联动封堵漏洞
java·网络·tcp/ip·github
ShiJiuD6668889992 小时前
大事件板块一
java
摇滚侠2 小时前
@Autowired 和 @Resource 的区别
java·开发语言
SeaTunnel2 小时前
(八)收官篇 | 数据平台最后一公里:数据集成开发设计与上线治理实战
java·大数据·开发语言·白鲸开源
吴声子夜歌3 小时前
Java——线程的基本协作机制
java·线程协作
谙弆悕博士3 小时前
【附C++源码】从零开始实现 2048 游戏
java·c++·游戏·源码·项目实战·2048
多加点辣也没关系4 小时前
设计模式-观察者模式
观察者模式·设计模式
独自归家的兔4 小时前
OCPP 1.6 协议详解:GetLocalListVersion 获取本地列表版本指令
java·后端·物联网·spring·ocpp1.6
hssfscv4 小时前
软件设计师下午题训练1-3题+2019上上午题错题解析 练习真题训练13
笔记·设计模式·uml