当前位置: 首页 > news >正文

设计模式系列(05):工厂方法模式(Factory Method)

本文为设计模式系列第5篇,聚焦创建型模式中的工厂方法模式,涵盖定义、原理、实际业务场景、优缺点、最佳实践及详细代码示例,适合系统学习与实战应用。


目录

  • 1. 模式概述
  • 2. 使用场景
  • 3. 优缺点分析
  • 4. 实际应用案例
  • 5. 结构与UML类图
  • 6. 代码示例
  • 7. 测试用例
  • 8. 常见误区与反例
  • 9. 最佳实践
  • 10. 参考资料与延伸阅读

1. 模式概述

工厂方法模式(Factory Method Pattern)是一种创建型设计模式。它通过定义一个用于创建对象的接口,让子类决定实例化哪一个类。工厂方法让类的实例化推迟到子类中进行,便于扩展和解耦。

1.1 定义

为创建对象定义一个接口,让子类决定实例化哪一个类。工厂方法使一个类的实例化延迟到其子类。

1.2 目的

  • 封装对象创建,隐藏实现细节
  • 支持扩展,便于新增产品类型
  • 解耦客户端与具体产品,提升灵活性

2. 使用场景

工厂方法模式在实际开发中应用广泛,常见场景包括:

  1. 日志系统:如支持多种日志输出(文件、数据库、远程等),通过工厂方法灵活切换日志实现。
  2. 数据库访问:如支持多种数据库(MySQL、Oracle、SQL Server等),通过工厂方法屏蔽底层差异。
  3. UI组件库:如跨平台UI库,工厂方法根据平台创建不同风格的控件。
  4. 文档处理:如支持多种文档格式(PDF、Word、Excel),通过工厂方法创建对应处理器。

真实业务背景举例:

  • 大型企业日志平台,需支持本地文件、数据库、远程服务器等多种日志存储方式,便于运维和审计。
  • SaaS平台根据客户定制不同UI风格,工厂方法可灵活切换控件实现。
  • 金融系统需对接多种数据库,工厂方法屏蔽底层实现,提升可维护性。

3. 优缺点分析

3.1 优点

  1. 封装性:隐藏对象创建细节,客户端只需依赖工厂接口。
  2. 扩展性:易于新增产品类型,符合开闭原则。
  3. 解耦性:客户端与具体产品解耦,便于维护和测试。

3.2 缺点

  1. 类数量增加:每新增一种产品都需对应工厂类,系统结构变复杂。
  2. 继承局限:扩展需继承工厂接口,可能导致继承层次过深。
  3. 使用限制:只适用于同族产品,不支持跨族产品创建。

4. 实际应用案例

  1. 日志系统:文件日志、数据库日志、远程日志等多种实现,便于扩展和切换。
  2. 数据库连接:根据配置创建不同数据库连接对象,屏蔽底层差异。
  3. UI控件工厂:根据操作系统或主题创建不同风格控件。
  4. 支付渠道对接:如微信、支付宝、银联等,每种渠道对应不同工厂。

5. 结构与UML类图

@startuml
package "Factory Method Pattern" #DDDDDD {interface Product {+ use(): void}class ConcreteProductA implements Productclass ConcreteProductB implements Productinterface Factory {+ createProduct(): Product}class ConcreteFactoryA implements Factoryclass ConcreteFactoryB implements Factoryclass ClientFactory <|.. ConcreteFactoryAFactory <|.. ConcreteFactoryBProduct <|.. ConcreteProductAProduct <|.. ConcreteProductBConcreteFactoryA --> ConcreteProductA : createProduct()ConcreteFactoryB --> ConcreteProductB : createProduct()Client ..> Factory : uses
}
@enduml

6. 代码示例

6.1 基本结构示例

业务背景: 企业日志平台需支持多种日志实现,便于灵活切换和扩展。

// 日志产品接口,定义日志操作
public interface Logger {/*** 写入日志信息* @param message 日志内容*/void log(String message);
}// 文件日志实现,写入本地文件
public class FileLogger implements Logger {private String filePath;public FileLogger(String filePath) {this.filePath = filePath;}@Overridepublic void log(String message) {// 实际业务:将日志写入指定文件try (FileWriter writer = new FileWriter(filePath, true)) {writer.write(message + "\n");} catch (IOException e) {e.printStackTrace();}}
}// 数据库日志实现,写入数据库表
public class DatabaseLogger implements Logger {private Connection connection;public DatabaseLogger(Connection connection) {this.connection = connection;}@Overridepublic void log(String message) {// 实际业务:将日志写入数据库try {PreparedStatement stmt = connection.prepareStatement("INSERT INTO logs (message, timestamp) VALUES (?, ?)");stmt.setString(1, message);stmt.setTimestamp(2, new Timestamp(System.currentTimeMillis()));stmt.executeUpdate();} catch (SQLException e) {e.printStackTrace();}}
}// 日志工厂接口,定义创建Logger的方法
public interface LoggerFactory {Logger createLogger();
}// 文件日志工厂,负责创建FileLogger
public class FileLoggerFactory implements LoggerFactory {private String filePath;public FileLoggerFactory(String filePath) {this.filePath = filePath;}@Overridepublic Logger createLogger() {return new FileLogger(filePath);}
}// 数据库日志工厂,负责创建DatabaseLogger
public class DatabaseLoggerFactory implements LoggerFactory {private Connection connection;public DatabaseLoggerFactory(Connection connection) {this.connection = connection;}@Overridepublic Logger createLogger() {return new DatabaseLogger(connection);}
}// 客户端示例
public class Client {public static void main(String[] args) throws Exception {// 创建文件日志工厂LoggerFactory fileFactory = new FileLoggerFactory("app.log");Logger fileLogger = fileFactory.createLogger();fileLogger.log("写入文件日志");// 创建数据库日志工厂Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/mydb", "user", "password");LoggerFactory dbFactory = new DatabaseLoggerFactory(conn);Logger dbLogger = dbFactory.createLogger();dbLogger.log("写入数据库日志");}// 总结:通过工厂方法,客户端无需关心Logger的具体实现,便于扩展和维护。
}

6.2 实际业务场景:支付渠道工厂

业务背景: 电商平台需对接多种支付渠道(如微信、支付宝、银联),通过工厂方法灵活扩展。

// 支付接口,定义支付操作
public interface Payment {void pay(double amount);
}// 微信支付实现
public class WeChatPayment implements Payment {@Overridepublic void pay(double amount) {System.out.println("[WeChat] 支付 " + amount + " 元");}
}
// 支付宝支付实现
public class AliPayPayment implements Payment {@Overridepublic void pay(double amount) {System.out.println("[AliPay] 支付 " + amount + " 元");}
}
// 银联支付实现
public class UnionPayPayment implements Payment {@Overridepublic void pay(double amount) {System.out.println("[UnionPay] 支付 " + amount + " 元");}
}
// 支付工厂接口
public interface PaymentFactory {Payment createPayment();
}
// 微信支付工厂
public class WeChatPaymentFactory implements PaymentFactory {public Payment createPayment() {return new WeChatPayment();}
}
// 支付宝支付工厂
public class AliPayPaymentFactory implements PaymentFactory {public Payment createPayment() {return new AliPayPayment();}
}
// 银联支付工厂
public class UnionPayPaymentFactory implements PaymentFactory {public Payment createPayment() {return new UnionPayPayment();}
}
// 客户端示例
public class PaymentClient {public static void main(String[] args) {PaymentFactory factory = new WeChatPaymentFactory();Payment payment = factory.createPayment();payment.pay(100.0);// 切换支付渠道只需更换工厂实现factory = new AliPayPaymentFactory();payment = factory.createPayment();payment.pay(200.0);}
}

7. 测试用例

业务背景: 验证日志工厂和支付工厂的多实现切换与功能正确性。

public class FactoryMethodPatternTest {@Testpublic void testFileLogger() {LoggerFactory factory = new FileLoggerFactory("test.log");Logger logger = factory.createLogger();logger.log("Test message");// 验证文件内容try {List<String> lines = Files.readAllLines(Paths.get("test.log"));assertTrue(lines.contains("Test message"));} catch (IOException e) {fail("Failed to read log file");}}@Testpublic void testDatabaseLogger() {Connection conn = createTestConnection();LoggerFactory factory = new DatabaseLoggerFactory(conn);Logger logger = factory.createLogger();logger.log("Test message");// 验证数据库记录try {PreparedStatement stmt = conn.prepareStatement("SELECT message FROM logs WHERE message = ?");stmt.setString(1, "Test message");ResultSet rs = stmt.executeQuery();assertTrue(rs.next());} catch (SQLException e) {fail("Failed to verify database log");}}@Testpublic void testPaymentFactory() {PaymentFactory factory = new WeChatPaymentFactory();Payment payment = factory.createPayment();assertTrue(payment instanceof WeChatPayment);factory = new AliPayPaymentFactory();payment = factory.createPayment();assertTrue(payment instanceof AliPayPayment);}private Connection createTestConnection() {try {return DriverManager.getConnection("jdbc:h2:mem:testdb;DB_CLOSE_DELAY=-1","sa","");} catch (SQLException e) {throw new RuntimeException("Failed to create test connection", e);}}
}

8. 常见误区与反例

  • 误区1:工厂类职责过重,变成"万能工厂"
    应按产品族拆分工厂,避免单一工厂膨胀。
  • 误区2:客户端仍直接依赖具体产品
    应通过工厂接口获取产品,避免高耦合。
  • 反例:工厂方法未能隐藏实现细节
    工厂应只暴露必要接口,隐藏创建逻辑。

9. 最佳实践

  1. 接口设计:工厂和产品接口要简洁,便于扩展和维护。
  2. 工厂拆分:按产品族或业务领域拆分工厂,避免"万能工厂"。
  3. 异常与资源管理:工厂方法应妥善处理异常和资源释放。
  4. 扩展性:新增产品类型时优先扩展工厂,不修改已有代码。
  5. 文档和UML同步:保持文档、UML和代码示例一致,便于团队协作。

10. 参考资料与延伸阅读

  • 《设计模式:可复用面向对象软件的基础》GoF
  • Effective Java(中文版)
  • https://refactoringguru.cn/design-patterns/factory-method
  • https://www.baeldung.com/java-factory-pattern

本文为设计模式系列第5篇,后续每篇将聚焦一个设计模式或设计原则,深入讲解实现与应用,敬请关注。

相关文章:

  • 传统工程项目管理与业财一体化管理的区别?
  • 高效获客利器:应用宝CPD广告的投放优化与流量质量保障
  • AI+制造:中小企业的低成本智能化转型
  • 智慧化工园区安全风险管控平台建设方案(Word)
  • 未来安全与持续进化
  • 二极管的等效电路
  • 【和春笋一起学C++】(十五)字符串作为函数参数
  • == 和 equals 的区别
  • 旧物回收小程序开发——让闲置再生价值,用科技撬动绿色经济
  • 如何把一台电脑作为另外一台电脑的显示器
  • 对比关系型数据库与NoSQL数据库
  • 【数组的定义数组与内存的关系】
  • 基于微信小程序的高校校园微活动管理系统设计与实现(源码+定制+开发)高校微信小程序校园活动发布与互动平台开发 面向大学生群体的校园活动移动平台设计与实现
  • Redis核心用法与通用命令全解析
  • 212. 单词搜索 II
  • 机器人收硬币算法c++,完整代码可运行
  • TDengine 高可用——双副本
  • Android Raspberry 请求 api 失败 iOS 请求成功【ssl 证书配置问题】
  • 【LangChain大模型应用与多智能体开发 ① 初识LangChain 】
  • 人工智能100问☞第30问:什么是损失函数?
  • 石家庄网站排名/网站改版公司哪家好
  • 怎么找网站开发公司/最好用的搜索神器
  • css3网站模板/产品线上推广方案
  • 兰州网站在哪备案/微信营销的成功案例
  • 人妖和美女做视频网站/直播代运营公司
  • 自建外贸推广网站有哪些/seo搜索优化是什么