一、工厂方法模式核心概念

工厂方法模式(Factory Method Pattern)是一种创建型设计模式,属于经典23种设计模式之一。其核心思想是:定义一个创建对象的接口,但将具体对象的实例化过程延迟到子类中实现。这种模式通过引入抽象层,将对象的创建与使用解耦,使系统更具扩展性。

关键角色解析

  1. 抽象产品(Product)
    定义产品的公共接口,所有具体产品必须实现这些方法。例如:

    from abc import ABC, abstractmethodclass Animal(ABC):@abstractmethoddef speak(self):pass
    
  2. 具体产品(ConcreteProduct)
    实现抽象产品接口的具体类,如DogCat

    class Dog(Animal):def speak(self):return "Woof!"class Cat(Animal):def speak(self):return "Meow!"
    
  3. 抽象工厂(Creator)
    声明工厂方法,返回抽象产品类型:

    class AnimalFactory(ABC):@abstractmethoddef create_animal(self):pass
    
  4. 具体工厂(ConcreteCreator)
    实现抽象工厂的接口,负责创建具体产品实例:

    class DogFactory(AnimalFactory):def create_animal(self):return Dog()class CatFactory(AnimalFactory):def create_animal(self):return Cat()
    

二、工厂方法模式vs简单工厂模式

特性工厂方法模式简单工厂模式
核心思想通过子类延迟实例化到具体工厂集中创建逻辑到单个工厂类
扩展性符合开闭原则,新增产品只需添加子类添加新产品需修改工厂类
复杂度类数量较多,系统复杂度较高结构简单,适合产品较少场景
适用场景产品族扩展频繁,需动态选择工厂产品类型固定,创建逻辑简单

三、Python实现工厂方法模式的三种方式

方式1:类方法工厂

利用@classmethod实现工厂方法,避免实例化工厂类:

class Animal:@classmethoddef factory(cls, animal_type):if animal_type == "dog":return Dog()elif animal_type == "cat":return Cat()else:raise ValueError("Invalid animal type")# 使用示例
animal = Animal.factory("dog")
print(animal.speak())  # 输出:Woof!

方式2:抽象基类强制实现

通过abc模块强制子类实现工厂方法:

from abc import ABC, abstractmethodclass AnimalFactory(ABC):@abstractmethoddef create(self):passclass DogFactory(AnimalFactory):def create(self):return Dog()# 客户端代码
factory = DogFactory()
animal = factory.create()

方式3:动态工厂映射

结合字典实现灵活映射关系:

class AnimalFactory:_factories = {"dog": Dog,"cat": Cat}@classmethoddef create(cls, animal_type):return cls._factories[animal_type]()# 使用示例
animal = AnimalFactory.create("cat")

四、工厂方法模式优缺点分析

✅ 优点

  1. 解耦创建与使用
    客户端无需知道具体类名,只需通过工厂接口获取对象。

  2. 符合开闭原则
    新增产品时,只需添加具体工厂和产品类,无需修改现有代码。

  3. 支持多态性
    不同工厂可返回同一接口的不同实现,便于替换实现逻辑。

❌ 缺点

  1. 类数量膨胀
    每新增一个产品需对应一个工厂类,增加系统复杂度。

  2. 抽象层引入
    过度使用可能导致代码可读性下降,需权衡设计复杂度。

五、实际应用场景

  1. 框架开发
    Django框架中的表单和模型字段创建:

    from django import formsclass CustomForm(forms.Form):name = forms.CharField()# 工厂方法动态生成字段
    def field_factory(field_type):if field_type == "text":return forms.CharField()elif field_type == "email":return forms.EmailField()
    
  2. 插件系统
    为不同插件提供统一创建接口:

    class Plugin:@abstractmethoddef load(self):passclass ImagePlugin(Plugin):def load(self):print("Loading image plugin")class VideoPlugin(Plugin):def load(self):print("Loading video plugin")class PluginFactory:@staticmethoddef create(plugin_type):if plugin_type == "image":return ImagePlugin()elif plugin_type == "video":return VideoPlugin()
    
  3. 日志系统
    根据配置动态创建日志处理器:

    class Logger:@abstractmethoddef log(self, message):passclass FileLogger(Logger):def log(self, message):with open("log.txt", "a") as f:f.write(message)class ConsoleLogger(Logger):def log(self, message):print(message)class LoggerFactory:@staticmethoddef get_logger(logger_type):if logger_type == "file":return FileLogger()else:return ConsoleLogger()
    

六、总结

工厂方法模式通过抽象工厂和具体工厂的分离,为对象创建提供了灵活的扩展机制。在Python中,可结合@classmethod、抽象基类(abc模块)或字典映射等方式实现。其核心价值在于将对象创建逻辑封装到独立模块,降低系统耦合度,特别适用于产品类型可能动态扩展的场景。

实践建议:当系统需要频繁添加新产品,且客户端不关心具体实现时,优先使用工厂方法模式;若产品类型固定且数量较少,可考虑简单工厂模式以降低复杂度。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/pingmian/89788.shtml
繁体地址,请注明出处:http://hk.pswp.cn/pingmian/89788.shtml
英文地址,请注明出处:http://en.pswp.cn/pingmian/89788.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python爬虫获取PDF

【前提:菜鸟学习的记录过程,如果有不足之处,还请各位大佬大神们指教(感谢)】 1.方法一:网站找到目标数据【单篇PDF】 https://bidding.sinopec.com/tpfront/xxgg/004005/ 按F12,----检查------…

IFN影视官网入口 - 4K影视在线看网站|网页|打不开|下载

IFN影视是一个专注于影视内容的网站,提供电影、电视剧、综艺等各类影视资源的在线观看服务。该网站以用户需求为导向,致力于为用户提供高清、流畅的观影体验,并不断更新内容以满足不同用户的观看习惯和偏好。IFN影视的特色在于其内容丰富、分…

《计算机网络》实验报告四 TCP协议分析

目 录 1、实验目的 2、实验环境 3、实验内容 3.1 利用wget下载新疆大学主页 3.2 使用wireshark分析TCP报文结构 3.3 使用wireshark分析建立连接的三次握手 3.4 使用wireshark分析释放连接的四次挥手 4、实验结果与分析 4.1 利用wget下载新疆大学主页 4.2 使用wiresh…

知识 IP 的突围:从 “靠感觉” 到 “系统 + AI” 的变现跃迁

越来越多的知识付费从业者陷入 “努力无成果” 的困局:做了内容、上了课程,却没人看、没人买。核心问题不在于能力不足,而在于仍在用 “靠感觉” 的原始方式打造 IP。在流量内卷、节奏加快的当下,“内容情怀” 已撑不起一门生意&a…

4.Java创建对象有几种方式?

1.使用 new 关键字(最常用)通过调用类的构造函数直接实例化对象Person person new Person(); // 调用无参构造 Person person new Person("Alice", 25); // 调用有参构造2.反射机制(动态创建)利用Java反射 API 在运行…

【好题】洛谷 P1600 [NOIP 2016 提高组] 天天爱跑步(倍增LCA+桶)

前言没做出来,看了很多篇题解后AC了,感觉大部分题解讲得不清楚。题目思路结果有两种求法模拟跑步过程,统计每个节点能观察到的人数考虑每条路径会对哪些节点作出贡献(当前路径的玩家能被观察到)尝试第一种求法必须遍历…

valkey之网络管理架构深度解析

一、连接类型实现体系 valkey通过ConnectionType结构体构建了灵活的网络连接抽象,支持多种连接类型的统一管理。每种连接类型都通过填充该结构体的函数指针来实现特定功能,形成了面向接口的设计模式。1.1 socket连接 Socket连接提供了最基础的TCP/IP通信…

【解码文本世界的“隐形分界线”:Windows与Linux回车换行之谜】

在计算机的文本世界里,回车(Carriage Return,CR)和换行(Line Feed,LF)是两个看似简单却意义非凡的字符。它们如同文本中的“隐形分界线”,默默地划分着段落与行,影响着文…

【Project】ELK 7.17.16 日志分析系统部署

ELK 日志分析系统集群部署 本文档基于 Rocky Linux 9.4 系统,部署 ELK 7.17.16(长期支持版)集群 案例准备 1. 节点规划IP主机名部署组件角色说明192.168.100.150kafka01Elasticsearch、Kibana主节点(master) 可视化192…

分布式定时任务系列13:死循环是任务触发的银弹?

传送门 分布式定时任务系列1:XXL-job安装 分布式定时任务系列2:XXL-job使用 分布式定时任务系列3:任务执行引擎设计 分布式定时任务系列4:任务执行引擎设计续 分布式定时任务系列5:XXL-job中blockingQueue的应用 …

Flutter基础(前端教程①③-单例)

现实类比:公司打印机假设你们公司有一台共享打印机:非单例(重复创建):每个员工都自己买一台打印机放在工位上结果:浪费钱,占空间,难维护单例(唯一实例)&#…

力扣刷题 -- 965.单值二叉树

题目示例: 思路分析代码实现 bool isUnivalTree(struct TreeNode* root) {if(rootNULL){return true;}if(root->left && root->val ! root->left->val){return false;}if(root->right && root->val ! root->right->val){re…

uni-api交互反馈组件(showToast)的用法

欢迎来到我的UniApp技术专栏!🎉 在这里,我将与大家分享关于UniApp开发的实用技巧、最佳实践和项目经验。 专栏特色: 📱 跨平台开发一站式解决方案 🚀 从入门到精通的完整学习路径 💡 实战项目经…

借助它,在Web3投资赛道抢占先机

随着互联网技术的飞速发展,Web3的概念逐渐成为科技圈和投资界的热门话题。Web3代表着下一代互联网的发展方向,它强调去中心化、用户主权和数据隐私保护。在这一新兴领域,如何借助Web3技术抢占投资先机,成为许多投资者关注的焦点。…

验证大语言模型不会算数但可以编写算数的程序

摘要:本文通过几个实例测试了大语言模型在数学计算、排序、统计等方面的能力。结果显示,对于简单字符统计、排序等任务,大模型能正确生成实现代码,但当数据区分度降低时容易出错。在计算学生分数排名任务中,大模型生成…

概率论与数理统计(八)

参数估计 通过取样本,并用样本构造函数,达成估计分布函数参数的目的 矩估计法 本质:用样本的各阶矩代替总体的各阶矩,即取: E(X)X‾1n∑iXiE(X2)1n∑iXi2E(X)\overline{X}\dfrac{1}{n}\sum_i X_i\\ E(X^2)\dfrac{1}…

服务器后台崩溃的原因

当我们双十一活动零点拼命刷新却卡在支付完页面,游戏页面等不进去,公司系统瘫痪全体员工干瞪眼,服务器崩溃绝对是数字时代中的酷刑!那服务器为什么会说崩就崩,用户对于这种情况该如何进行避雷呢?服务器主要…

线程池与ThreadPoolExecutor源码解析(上)

一、线程池线程池(ThreadPool)是一种线程复用的机制。它维护着若干个线程,任务来了就复用这些线程去执行,任务做完线程不会销毁,而是回到池中等待下一个任务。为什么要用线程池?降低资源消耗:避…

Linux内核IP分片重组机制剖析:高效与安全的艺术

在IP网络通信中,当数据包超过MTU限制时,路由器会将其拆分为多个分片。这些分片到达目标主机后,内核必须高效、安全地重组原始数据包。Linux内核的net/ipv4/inet_fragment.c实现了一套精妙的分片管理框架,完美平衡了性能和安全性需求。本文将深入剖析其设计哲学与关键技术。…

相机模型和对极几何

一、相机模型 1.针孔相机模型-外参矩阵 1.世界坐标系到相机坐标系 世界坐标系:可以定义空间中任意一个位置,原点位置三个坐标轴方向坐标系姿态(X,Y,Z)相机坐标系:定义在相机上,原点是相机中心,z…