设计模式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版》专栏!关注不迷路~

相关推荐
mahuifa4 分钟前
OpenCV 开发 -- 图像基本处理
人工智能·python·opencv·计算机视觉
一个java开发39 分钟前
distributed.client.Client 用户可调用函数分析
大数据·python
eqwaak01 小时前
Matplotlib 动态显示详解:技术深度与创新思考
网络·python·网络协议·tcp/ip·语言模型·matplotlib
007php0071 小时前
某大厂MySQL面试之SQL注入触点发现与SQLMap测试
数据库·python·sql·mysql·面试·职场和发展·golang
CodeCraft Studio1 小时前
Excel处理控件Aspose.Cells教程:使用 Python 将 Pandas DataFrame 转换为 Excel
python·json·excel·pandas·csv·aspose·dataframe
flashlight_hi1 小时前
LeetCode 分类刷题:2563. 统计公平数对的数目
python·算法·leetcode
java1234_小锋1 小时前
Scikit-learn Python机器学习 - 特征预处理 - 归一化 (Normalization):MinMaxScaler
python·机器学习·scikit-learn
星空的资源小屋2 小时前
网易UU远程,免费电脑远程控制软件
人工智能·python·pdf·电脑
IMER SIMPLE2 小时前
人工智能-python-深度学习-神经网络-MobileNet V1&V2
人工智能·python·深度学习
eleqi2 小时前
Python+DRVT 从外部调用 Revit:批量创建楼板
python·系统集成·revit·外部调用·drvt·自动化生产流水线