python设计模式-工厂模式

工厂模式的核心思想:封装对象创建过程、解耦对象使用与创建 。

示例代码:

python 复制代码
from enum import Enum


# 基类:人类
class Person:
    species = 'Homo sapiens'

    def __init__(self, name):
        self.name = name

    def __str__(self):
        return f"{self.__class__.__name__}: {self.name}"


# 子类:成年人
class Adult(Person):
    def can_vote(self):
        return True  # 成年人有投票权


# 子类:未成年人
class Minor(Person):
    def can_vote(self):
        return False  # 未成年人无投票权


# 子类:匿名人士
class AnonymousPerson(Person):
    def __init__(self):
        super().__init__('Anonymous')  # 固定名称为Anonymous


# 枚举:定义支持的人物类型
class PersonType(Enum):
    ADULT = 1
    MINOR = 2
    ANONYMOUS = 3


# 工厂类:负责创建不同类型的Person对象
class PersonFactory:
    @staticmethod
    def create_person(person_type, name=None):
        """
        根据类型创建不同的Person对象
        :param person_type: PersonType枚举值,指定要创建的对象类型
        :param name: 名称(匿名类型不需要此参数)
        :return: 对应的Person子类实例
        """
        if person_type == PersonType.ADULT:
            if not name:
                raise ValueError("创建成年人需要提供姓名")
            return Adult(name)
        elif person_type == PersonType.MINOR:
            if not name:
                raise ValueError("创建未成年人需要提供姓名")
            return Minor(name)
        elif person_type == PersonType.ANONYMOUS:
            return AnonymousPerson()  # 匿名类型无需姓名
        else:
            raise ValueError(f"不支持的人物类型: {person_type}")


# 使用示例
if __name__ == "__main__":
    # 通过工厂创建不同类型的对象
    adult = PersonFactory.create_person(PersonType.ADULT, "Alice")
    minor = PersonFactory.create_person(PersonType.MINOR, "Bob")
    anonymous = PersonFactory.create_person(PersonType.ANONYMOUS)

    print(adult)  # 输出:Adult: Alice
    print(minor)  # 输出:Minor: Bob
    print(anonymous)  # 输出:AnonymousPerson: Anonymous

    print(adult.can_vote())  # 输出:True
    print(minor.can_vote())  # 输出:False

以上代码通过创建独立的工厂类,并通过工厂类来管理不同类型对象的创建。

这个实现的特点(符合标准工厂模式):

  1. 分离的工厂类

    创建了独立的PersonFactory类,专门负责对象的创建逻辑,而Person及其子类只关注自身的业务逻辑(如can_vote方法)。这种分离符合 "单一职责原则"。

  2. 通过类型控制创建

    使用PersonType枚举明确支持的对象类型,调用者只需指定类型和必要参数(如姓名),无需直接实例化具体子类(如Adult()Minor())。

  3. 封装创建细节

    工厂类内部处理了不同类型对象的创建条件(如匿名对象不需要姓名),调用者无需关心这些细节,只需通过统一的create_person方法获取对象。

  4. 易于扩展

    如果需要新增Person类型(如Senior老年人),只需:

    • 创建Senior子类继承Person
    • PersonType枚举中添加SENIOR
    • 在工厂类的create_person方法中增加对应分支
      无需修改现有业务逻辑,符合 "开闭原则"。
  5. 统一的接口

    所有通过工厂创建的对象都遵循Person基类的接口规范(如都有name属性和__str__方法),调用者可以统一处理这些对象,无需区分具体类型。

这种实现比之前的类方法方式更接近标准工厂模式,尤其在需要管理多种对象类型、创建逻辑复杂的场景下,优势会更加明显。

相关推荐
小徐不徐说6 分钟前
C++ 模板与 STL 基础入门:从泛型编程到实战工具集
开发语言·数据结构·c++·qt·面试
艾莉丝努力练剑8 分钟前
【C/C++】类和对象(上):(一)类和结构体,命名规范——两大规范,新的作用域——类域
java·c语言·开发语言·c++·学习·算法
AndrewHZ20 分钟前
【图像处理基石】如何对遥感图像进行实例分割?
图像处理·人工智能·python·大模型·实例分割·detectron2·遥感图像分割
froginwe1124 分钟前
WebPages PHP:深入解析PHP在网页开发中的应用
开发语言
No0d1es36 分钟前
第13届蓝桥杯Python青少组中/高级组选拔赛(STEMA)2021年11月27日真题
python·青少年编程·蓝桥杯·选拔赛
天天找自己1 小时前
精通分类:解析Scikit-learn中的KNN、朴素贝叶斯与决策树(含随机森林)
python·决策树·机器学习·分类·scikit-learn
R-G-B1 小时前
【33】C# WinForm入门到精通 ——表格布局器TableLayoutPanel【属性、方法、事件、实例、源码】
开发语言·c#·c# winform·表格布局器·tablelayoutpane
郝学胜-神的一滴1 小时前
Spring Boot Actuator 保姆级教程
java·开发语言·spring boot·后端·程序人生
赵英英俊1 小时前
Python day31
开发语言·python
硬核子牙2 小时前
Python虚拟机内存机制底层
python