Python 设计模式之工厂函数模式
文章目录
- 案例
- 基本案例
- 逐渐复杂的案例
- 问题回顾
- 什么是工厂模式?
- 为什么会用到工厂函数模式?
- 工厂函数模式和抽象工厂模式有什么关系?
工厂函数模式是一种创建型设计模式,抛出问题:
- 什么是工厂函数模式?
- 为什么会用到工厂函数模式?
- 工厂函数模式和抽象工厂模式有什么关系?
案例
基本案例
为了理解这些问题,需要引入一个场景案例:
一个城市的公共交通一开始只有公交车,该城市通过调用公交车解决城市人流的流动:
def transfer(people):print(f"move {people} mans to other place")def main():people = 100transfer(people)
随着城市的发展,开始支持地铁,并且这些交通方式也有了复杂的定义,于是新的版本:
class Bus:def transfer(self, people):print(f"move {people} mans to other place on Road")class Subway:def transfer(self, people):print(f"move {people} mans to other place under ground")def main():people = 100w = Bus()w.transfer(people)main()
这里我们利用打印语句内容的不同来暗示它们的行为不同,而 main
函数作为客户端代码只需要通过调用不同的交通方式类来得到实例对象,再调用具体方法即可达成目的。假如还有其他交通方式引入,那就再实现多一个类就可以了。这个版本到这里并没有什么不好。
这个城市后来又发展出轮渡的交通方式,同时客户端代码的逻辑也变得更加复杂,对交通方式的实例类的调用也更频繁,原来只是调用 w.transfer(people)
,现在还需要执行载货、或者设备报告等等方式,这意味着每次引入一个新的交通方式类,就要保证这个新的类的引入不会导致客户端的代码不会出错。
- 新的类没有实现的方法,而客户端代码中在后面调用了
- 有些行为只有某一个产品才有
- 新的维护者用其他方式命名了该方法
为了规避这些问题,开发者对这些交通方式(产品)进行了抽象(抽象产品),任何新引入的交通方式必须继承这个抽象产品,并且实现这个抽象产品中的基本方法,于是这个类关系图变成:
而据此实现的新版本:
from abc import ABC, abstractmethodclass Product(ABC):@abstractmethoddef transfer(self, people):...class Bus(Product):def transfer(self, people):print(f"move {people} mans to other place on Road")class Subway(Product):def transfer(self, people):print(f"move {people} mans to other place under ground")class Ship(Product):def transfer(self, people):print(f"move {people} mans to other place on sea")def main():people = 100w = Ship()w.transfer(people)main()
在这种方式下,如果继承自 Product
的交通方式不实现其定义的抽象接口就无法实例化。进行到这里实际上关于工厂函数的四个角色,我们已经引出了两个,即下面标红的抽象产品
和具体产品
,指代我们的交通方式
和具体交通方式
- Creator
- Concrete Creator
Product
Concrete Product
逐渐复杂的案例
让我们继续给这个上面的案例加多一些现实场景:
随着发展,每一种具体的交通方式都有了各自复杂多样的类型(具备不同属性的实例)。以前只有一辆24座的红色巴士,现在城市发展了,有50座红色巴士、50座绿色巴士、使用新能源的、使用汽油的…这样的情况在地铁和轮船上也是如此。
而针对运输一批市民的需求,也要根据实际情况不同选择合适的产品,就算是用巴士运载,也要选用合适座位数等属性的大巴车进行调度。
这体现在我们的代码中是怎样的呢?以 Bus 为例:
class Bus(Product):def __init__(self, seat, color, energy):self.color = colorself.seat = seatself.energy = energydef transfer(self, people):print(f"move {people} mans to other place on Road")def main():people = 100if people < 24:w = Bus(color="red", seat=24, energy="电")else:w = Bus(color="red", seat=120, energy="油")w.transfer(people)main()
可以看到,客户端代码需要根据情景不同生成不同属性的实例,这还只是考虑了一个属性(人数)的逻辑,如果还需根据其他基本信息,那客户端代码就更复杂了,比如:
- 目的地不同选择电还是油?
- 同样是人数小于 24人时,生成轮船所需的配置参数和大巴的配置参数不同
这些问题如果都交给客户端去解决的话,那就会导致:
- 一旦引入新的产品就要修改一次代码
- 客户端需要负责实例的生成,高度的耦合
那么可不可以客户端只负责提出:
- 期望的产品(实例类型:大巴还是轮船)
- 问题的基本信息(人数、目的地…)即生成实例的基本参数
这样的背景下,我们引入创建者角色,由创建者根据不同的入参创建不同属性配置的实例
而轮船也有其对应的轮船创建者、地铁对应其地铁创建者,这类创建者多了之后就衍生了规范,每一种创建者需要遵循某种基本规范:
讲到这里,我们就引入了抽象创建者
和具体创建者
, 到此工厂函数模式的四个角色就都到齐了。让我们把这新出现的两个角色也整合进代码中:
from abc import ABC, abstractmethodclass Product(ABC):@abstractmethoddef transfer(self, people):...class Bus(Product):def __init__(self, seat, color, energy):self.color = colorself.seat = seatself.energy = energydef transfer(self, people):print(f"move {people} mans to other place on Road")class Subway(Product):def transfer(self, people):print(f"move {people} mans to other place under ground")class Ship(Product):def transfer(self, people):print(f"move {people} mans to other place on sea")class Creator(ABC):@abstractmethoddef create_transportation(self):...class BusCreator(Creator):def __init__(self, people):self.expected_people = peopledef create_transportation(self):if self.expected_people < 24:return Bus(seat=24,color="red", energy="油")return Bus(seat=120,color="yellow", energy="电")class ShipCreator(Creator):def create_transportation(self):return Ship()def main():people = 10creator = BusCreator(people)w = creator.create_transportation()w.transfer(people)print(w.seat)main()
可以看到复杂的实例生成逻辑已经被我们从 main
函数即客户端代码中抽离出来,而如果使用者希望轮船这个产品,他只需更换为对应的创建者即可,即使轮船实例有其特殊的创建逻辑也无需考虑。
现在我们这段代码的类关系图如下:
客户端代码不直接调用具体产品类(Bus
)来得到具体实例 Bus instance
,而是通过调用具体创建者(BusCreator
)来获得具体实例。
问题回顾
现在回过头来看博客开头提到的三个问题
什么是工厂模式?
模式定义
在工厂方法模式中,工厂父类负责定义创建产品对象的公共接口,而工厂子类则负责生成具体的产品对象,这样做的目的是将产品类的实例化操作延迟到工厂子类中完成,即通过工厂子类来确定究竟应该实例化哪一个具体产品类。
这里的工厂父类是指抽象创建者
Creator
, 它提供了一个需要被实现的抽象方法creat_transportation
, 而让其子类在这个函数里实现实例化对象时的定制逻辑
工厂方法模式建议使用特殊的工厂方法代替对于对象构造函数的直接调用。而上面的案例用具体创建者的函数
creat_transportation
(工厂方法) 代替直接使用Bus()
(即客户端直接使用Bus.__init__()
)
模式结构
工厂方法模式包含如下角色:
- Product:抽象产品
- ConcreteProduct:具体产品
- Factory:抽象工厂
- ConcreteFactory:具体工厂
这四种角色,上面在案例中已经做了拆解就不做赘述。
为什么会用到工厂函数模式?
在上面的案例演变中,我觉得你应该得到这个问题的答案。如果没有,那么概括下:
- 如果这是一个简单的对象,不涉及复杂的实例创建过程,那么不需要应用这个模式,反过来讲,使用工厂函数模式是为了解决在客户端代码需要根据
具体场景
创建具体的复杂
实例的问题。
工厂函数模式和抽象工厂模式有什么关系?
到点吃饭了,这个问题暂时先不展开论述了,其实你仔细看上面的代码可以发现有部分抽象工厂模式的思想。
- 虽然代码示例中没有直接展示出抽象工厂模式的完整结构,但可以看到
Product
是一个抽象类,定义了产品的抽象接口transfer()
,而具体的产品类Bus
、Subway
和Ship
实现了这个接口。 Creator
类可以被视为一个抽象工厂,定义了一个创建产品的抽象方法create_transportation()
,而BusCreator
和ShipCreator
则是具体工厂,实现了 Creator 接口并负责创建具体产品。
关系和区别
- 关系:工厂函数模式可以被看作是抽象工厂模式的一种简化形式,特别是在只需创建单一种类的情况下。它们都用于将对象的创建与使用代码分离,提高系统的可扩展性和灵活性。
- 区别:主要区别在于抽象工厂模式更加抽象和灵活,能够创建
一组相关对象
(产品家族),而工厂函数模式通常只涉及单一对象类型
的创建。抽象工厂模式还涉及多个抽象类或接口的定义和实现,更适用于需要创建多个相关对象的场景。
相关文章:

Python 设计模式之工厂函数模式
文章目录 案例基本案例逐渐复杂的案例 问题回顾什么是工厂模式?为什么会用到工厂函数模式?工厂函数模式和抽象工厂模式有什么关系? 工厂函数模式是一种创建型设计模式,抛出问题: 什么是工厂函数模式?为什么…...
数据赋能(171)——开发:数据挖掘——概述、关注焦点
概述 数据挖掘是从大量的数据中,提取隐藏在其中的、事先不知道的、但潜在有用的信息的过程。 数据挖掘是数据分析过程中的一个核心环节。 数据挖掘的主要目的是从大量数据中自动发现隐藏的模式、关联和趋势,以揭示数据的潜在价值。数据挖掘技术可以帮…...

L1 - OpenCompass 评测 InternLM-1.8B 实践
基础任务(完成此任务即完成闯关) 使用 OpenCompass 评测 internlm2-chat-1.8b 模型在 ceval 数据集上的性能,记录复现过程并截图。 按照教程中的顺序安装包有问题,网上找了解决方案,按一下顺序能正常执行 使用OpenCo…...
JS【详解】数据类型检测(含获取任意数据的数据类型的函数封装、typeof、检测是否为 null、检测是否为数组、检测是否为非数组/函数的对象)
【函数封装】获取任意数据的数据类型 /*** 获取任意数据的数据类型** param x 变量* returns 返回变量的类型名称(小写字母)*/ function getType(x) {// 获取目标数据的私有属性 [[Class]] 的值const originType Object.prototype.toString.call(x); //…...

OpenCV图像滤波(10)Laplacian函数的使用
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 功能描述 计算图像的拉普拉斯值。 该函数通过使用 Sobel 运算符计算出的 x 和 y 的二阶导数之和来计算源图像的拉普拉斯值: dst Δ src ∂…...

docker系列11:Dockerfile入门
传送门 docker系列1:docker安装 docker系列2:阿里云镜像加速器 docker系列3:docker镜像基本命令 docker系列4:docker容器基本命令 docker系列5:docker安装nginx docker系列6:docker安装redis docker系…...

LVS(Linux virual server)详解
目录 一、LVS(Linux virual server)是什么? 二、集群和分布式简介 2.1、集群Cluster 2.2、分布式 2.3、集群和分布式 三、LVS运行原理 3.1、LVS基本概念 3.2、LVS集群的类型 3.2.1 nat模式 3.2.2 DR模式 3.2.3、LVS工作模式总结 …...
Session共享方法
在Web开发中,会话(Session)管理是跟踪用户与服务器之间交互的一种常见方法。Session 共享通常指的是在一个应用集群或多个应用服务之间保持用户的会话状态一致。这在负载均衡、微服务架构或者分布式系统中尤为重要 一、基于SQL的session管理…...

Ubuntu 22.04 Docker安装笔记
1、准备一台虚机 可以根据《VMware Workstation安装Ubuntu 22.04笔记》来准备虚拟机。完成后,根据需求安装必要的软件,并设置root权限进行登录。 sudo apt update sudo apt install iputils-ping -y sudo apt install vim -y允许root ssh登录࿱…...
编程-设计模式 6:适配器模式
设计模式 6:适配器模式 定义与目的 定义:适配器模式将一个类的接口转换成客户希望的另一个接口。适配器模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。目的:该模式的主要目的是解决接口不匹配的问题,使得一个…...
ERC721 概念解释
目录 FeaturesVotesAccess ControlUpgradeabilityFeatures Mintable: 允许创建新的代币(minting)。合约的所有者或有权限的账户可以调用 mint 函数来生成新的代币,并将其分配给指定的地址。 Auto Increment Ids:自动递增 ID。每次创建新的代币时,代币的 ID 会自动递增,确保…...
数据结构(其五)--串
目录 12.串 12.1 基本操作 12.2 串的存储结构 12.3 字符串的模式匹配算法 (1).朴素模式匹配算法 (2).KMP算法 i.next[]数组的求解 ii.next[]数组的优化——nextval数组 iii.手算nextval数组 iiii.机算nextval数组 + KMP函数 12.串 串,即字符串(string),由零个或多…...
LeetCode Hot100 LRU缓存
请你设计并实现一个满足 LRU (最近最少使用) 缓存 约束的数据结构。 实现 LRUCache 类: LRUCache(int capacity) 以 正整数 作为容量 capacity 初始化 LRU 缓存int get(int key) 如果关键字 key 存在于缓存中,则返回关键字的值,否则返回 -…...

GESP C++ 2024年06月一级真题卷
一、单选题(每题 2 分,共 30 分) 第 1 题 在 C 中,下列不可做变量的是 ( ) 。 A. five-Star B. five_star C. fiveStar D. _fiveStar 答案:A 解析:标识符命名规则,标识符由字母、数…...
在 Ubuntu Server 上配置静态 IP 地址
在 Ubuntu Server 上配置静态 IP 地址 测试时使用的Ubuntu server版本是22.04 一、Ubuntu 17.10之前版本 使用 ifupdown 配置文件来设置静态 IP。配置文件通常位于 /etc/network/interfaces。 1.1 编辑 /etc/network/interfaces 文件: sudo vim /etc/network/in…...

数据结构——栈的讲解(超详细)
前言: 小编已经在前面讲完了链表和顺序表的内容,下面我们继续乘胜追击,开始另一个数据结构:栈的详解,下面跟上小编的脚步,开启今天的学习之路! 目录 1.栈的概念和结构 1.1.栈的概念 1.2.栈的结构…...

三防平板助力MES系统,实现工厂移动式生产报工
在当今竞争激烈的制造业环境中,提高生产效率、优化生产流程以及实现精准的生产管理已经成为企业生存和发展的关键。 MES系统作为连接企业计划层和控制层的桥梁,在实现生产过程的信息化、数字化和智能化方面发挥着重要作用。与此同时,三防平板…...
WEB渗透Bypass篇-常规函数绕过
常规函数绕过 <?php echo exec(whoami);?> ------------------------------------------------------ <?php echo shell_exec(whoami);?> ------------------------------------------------------ <?php system(whoami);?> ------------------------…...

C++从入门到起飞之——string类的模拟实现 全方位剖析!
🌈个人主页:秋风起,再归来~🔥系列专栏:C从入门到起飞 🔖克心守己,律己则安 目录 1、多文件之间的关系 2、模拟实现常用的构造函数 2.1 无参构造函数 2.2 有参的构造函数 2.3 析构函…...
数据库国产化大趋势下,还需要学习Oracle吗?
由于众所周知的原因,近两年各行各业都开始了数据库国产化替代的进程,从国外商业数据库替换到国产或者开源数据库,相信很多的数据库从业人员会把部分精力转移到其他数据库产品的学习中,也有一些人在大肆的宣扬Oracle已经过时了&…...
变量 varablie 声明- Rust 变量 let mut 声明与 C/C++ 变量声明对比分析
一、变量声明设计:let 与 mut 的哲学解析 Rust 采用 let 声明变量并通过 mut 显式标记可变性,这种设计体现了语言的核心哲学。以下是深度解析: 1.1 设计理念剖析 安全优先原则:默认不可变强制开发者明确声明意图 let x 5; …...

手游刚开服就被攻击怎么办?如何防御DDoS?
开服初期是手游最脆弱的阶段,极易成为DDoS攻击的目标。一旦遭遇攻击,可能导致服务器瘫痪、玩家流失,甚至造成巨大经济损失。本文为开发者提供一套简洁有效的应急与防御方案,帮助快速应对并构建长期防护体系。 一、遭遇攻击的紧急应…...
pam_env.so模块配置解析
在PAM(Pluggable Authentication Modules)配置中, /etc/pam.d/su 文件相关配置含义如下: 配置解析 auth required pam_env.so1. 字段分解 字段值说明模块类型auth认证类模块,负责验证用户身份&am…...

基于当前项目通过npm包形式暴露公共组件
1.package.sjon文件配置 其中xh-flowable就是暴露出去的npm包名 2.创建tpyes文件夹,并新增内容 3.创建package文件夹...
什么?连接服务器也能可视化显示界面?:基于X11 Forwarding + CentOS + MobaXterm实战指南
文章目录 什么是X11?环境准备实战步骤1️⃣ 服务器端配置(CentOS)2️⃣ 客户端配置(MobaXterm)3️⃣ 验证X11 Forwarding4️⃣ 运行自定义GUI程序(Python示例)5️⃣ 成功效果
图表类系列各种样式PPT模版分享
图标图表系列PPT模版,柱状图PPT模版,线状图PPT模版,折线图PPT模版,饼状图PPT模版,雷达图PPT模版,树状图PPT模版 图表类系列各种样式PPT模版分享:图表系列PPT模板https://pan.quark.cn/s/20d40aa…...

Mac下Android Studio扫描根目录卡死问题记录
环境信息 操作系统: macOS 15.5 (Apple M2芯片)Android Studio版本: Meerkat Feature Drop | 2024.3.2 Patch 1 (Build #AI-243.26053.27.2432.13536105, 2025年5月22日构建) 问题现象 在项目开发过程中,提示一个依赖外部头文件的cpp源文件需要同步,点…...

Linux 内存管理实战精讲:核心原理与面试常考点全解析
Linux 内存管理实战精讲:核心原理与面试常考点全解析 Linux 内核内存管理是系统设计中最复杂但也最核心的模块之一。它不仅支撑着虚拟内存机制、物理内存分配、进程隔离与资源复用,还直接决定系统运行的性能与稳定性。无论你是嵌入式开发者、内核调试工…...
C#学习第29天:表达式树(Expression Trees)
目录 什么是表达式树? 核心概念 1.表达式树的构建 2. 表达式树与Lambda表达式 3.解析和访问表达式树 4.动态条件查询 表达式树的优势 1.动态构建查询 2.LINQ 提供程序支持: 3.性能优化 4.元数据处理 5.代码转换和重写 适用场景 代码复杂性…...
【Android】Android 开发 ADB 常用指令
查看当前连接的设备 adb devices 连接设备 adb connect 设备IP 断开已连接的设备 adb disconnect 设备IP 安装应用 adb install 安装包的路径 卸载应用 adb uninstall 应用包名 查看已安装的应用包名 adb shell pm list packages 查看已安装的第三方应用包名 adb shell pm list…...