设计模式Python版 工厂方法模式

文章目录


前言

GOF设计模式分三大类:

  • 创建型模式:关注对象的创建过程,包括单例模式、简单工厂模式、工厂方法模式、抽象工厂模式、原型模式和建造者模式。
  • 结构型模式:关注类和对象之间的组合,包括适配器模式、桥接模式、组合模式、装饰模式、外观模式、享元模式和代理模式。
  • 行为型模式:关注对象之间的交互,包括职责链模式、命令模式、解释器模式、迭代器模式、中介者模式、备忘录模式、观察者模式、状态模式、策略模式、模板方法模式和访问者模式。

一、工厂方法模式

工厂方法模式(Factory Method Pattern)

  • 定义:工厂方法模式提供一个抽象工厂接口来声明抽象工厂方法,而由其子类来具体实现工厂方法,创建具体的产品对象。客户端针对抽象工厂编程,可在运行时再指定具体工厂类。

  • 解决问题:如何通过不同的工厂来创建不同类型的对象?(每个具体工厂只生产一个具体产品)

  • 使用场景:

    • 与简单工厂模式相比,新增产品时只需要增加新的具体产品和具体工厂类,不需要修改已有代码,符合开闭原则
    • 创建对象的过程需要根据上下文环境变化,或者一个类不知道它所创建的对象的类
    • 一个类希望由其子类来指定创建的对象,或者系统需要通过子类来扩展
  • 具体场景:

    • 日志记录器:根据不同的日志级别(如DEBUG、INFO、ERROR)来创建不同的日志记录器。

    • 数据库访问:根据不同的数据库类型(如MySQL、Oracle、SQLite)来创建不同的数据库访问对象。

    • 支付网关:根据不同的支付方式(如信用卡、PayPal、支付宝)来创建不同的支付处理器。

    • 文件解析器:根据不同的文件类型(如PDF、Word、Excel)来创建不同的文件解析器。

    • UI组件:在图形用户界面应用程序中,根据不同的操作系统(如Windows、Mac、Linux)来创建不同的UI组件。

  • 组成:

    • 抽象产品(Product):定义产品的接口
    • 具体产品(Concrete Product):实现了抽象产品接口的具体类。
    • 抽象工厂(Creator):声明工厂方法,该方法返回一个产品类型的对象
    • 具体工厂(Concrete Creator):定义工厂方法以返回一个具体产品类的实例。
  • 优点:

    • 良好的扩展性。
    • 工厂方法模式是使用频率最高的设计模式之一,是很多开源框架和API类库的核心模式。
  • 缺点:

    • 系统中类的个数成对增加,在一定程度上增加了系统的复杂度

二、工厂方法模式示例

使用工厂方法模式来设计日志记录器

python 复制代码
# 模块 loggers.py
class Logger:
    """抽象产品"""

    def write_log(self, msg: str):
        raise NotImplementedError


class FileLogger(Logger):
    """具体产品"""

    def write_log(self, msg):
        print(f"文件日志记录:{msg}")


class DatabaseLogger(Logger):
    def write_log(self, msg):
        print(f"数据库日志记录:{msg}")


class LoggerFactory:
    """抽象工厂"""

    def create_logger(self) -> Logger:
        raise NotImplementedError


class FileLoggerFactory(LoggerFactory):
    """具体工厂"""

    def create_logger(self):
        # 创建文件等操作(略)
        return FileLogger()


class DatabaseLoggerFactory(LoggerFactory):
    def create_logger(self):
        # 连接数据库等操作(略)
        return DatabaseLogger()


# 客户端代码
factory = FileLoggerFactory()
logger = factory.create_logger()
logger.write_log('[22/Jan/2025 11:24:49] "GET /admin/ HTTP/1.1" 302 0')

三、工厂方法模式客户端改进

反射与配置文件:通过读取配置文件获取类名字符串,再使用反射机制,根据类名字符串生成对象。

  • 配置文件config.json
json 复制代码
{
    "class_name": "DatabaseLoggerFactory"
}
python 复制代码
from pathlib import Path
import json

class JsonUtil:
    @staticmethod
    def get_class_name():
        """读取配置文件,返回配置文件中的配置"""
        path = Path("config.json")
        contents = path.read_text(encoding="utf-8")
        conf = json.loads(contents)
        return conf.get("class_name", None)
python 复制代码
import loggers
from utils import JsonUtil

class_name = JsonUtil.get_class_name()
klass = getattr(loggers, class_name)
factory: loggers.LoggerFactory = klass()
logger = factory.create_logger()
logger.write_log('[22/Jan/2025 11:24:49] "GET /admin/ HTTP/1.1" 302 0')


### 输出结果
数据库日志记录:[22/Jan/2025 11:24:49] "GET /admin/ HTTP/1.1" 302 0

四、工厂方法模式隐藏工厂方法(可选)

通过将业务方法的调用移入工厂类,可以直接使用工厂对象来调用产品对象的业务方法,客户端无须直接使用工厂方法

python 复制代码
class LoggerFactory:
    """抽象工厂"""

    def write_log(self, msg: str):
        raise NotImplementedError


class FileLoggerFactory(LoggerFactory):
    """具体工厂"""

    def __init__(self):
        self.logger = FileLogger()

    def write_log(self, msg):
        self.logger.write_log(msg)


# 客户端代码
factory = FileLoggerFactory()
factory.write_log('[22/Jan/2025 11:24:49] "GET /admin/ HTTP/1.1" 302 0')

您正在阅读的是《设计模式Python版》专栏!关注不迷路~

相关推荐
兵慌码乱8 小时前
基于 MediaPipe 与 PySide2 的手势交互音乐控制系统实现:轻量化视觉交互全流程解析
python·opencv·计算机视觉·人机交互·手势识别·mediapipe·pyside2
luckdewei10 小时前
FastAPI 资产管理系统实战:复杂 ORM 关联、Alembic 迁移与 N+1 查询优化
python
用户69190268133911 小时前
Vibe Coding 开发项目的基本范式
人工智能·设计模式·代码规范
aqi0016 小时前
15天学会AI应用开发(八)使用向量数据库实现RAG功能
人工智能·python·大模型·ai编程·ai应用
Csvn17 小时前
`functools.lru_cache` —— 一行代码搞定缓存加速
后端·python
金銀銅鐵1 天前
[Python] 从《千字文》中随机挑选汉字
后端·python
怕浪猫1 天前
领域特定语言(Domain-Specific Language, DSL)
设计模式·程序员·架构
cup112 天前
[技术复盘] Windows Python 打包实战:Nuitka 环境踩坑总结与 CI 自动化构建全指南
python·ai·环境变量·ci·nuitka·skill