Python 设计模式之工厂函数模式

news2024/11/13 19:21:17

文章目录

    • 案例
      • 基本案例
      • 逐渐复杂的案例
    • 问题回顾
      • 什么是工厂模式?
      • 为什么会用到工厂函数模式?
      • 工厂函数模式和抽象工厂模式有什么关系?

工厂函数模式是一种创建型设计模式,抛出问题:

  1. 什么是工厂函数模式?
  2. 为什么会用到工厂函数模式?
  3. 工厂函数模式和抽象工厂模式有什么关系?

案例

基本案例

为了理解这些问题,需要引入一个场景案例:

一个城市的公共交通一开始只有公交车,该城市通过调用公交车解决城市人流的流动:

def transfer(people):
    print(f"move {people} mans to other place")

def main():
    people = 100
    transfer(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 = 100
    w = Bus()
    w.transfer(people)

main()

这里我们利用打印语句内容的不同来暗示它们的行为不同,而 main 函数作为客户端代码只需要通过调用不同的交通方式类来得到实例对象,再调用具体方法即可达成目的。假如还有其他交通方式引入,那就再实现多一个类就可以了。这个版本到这里并没有什么不好。
这个城市后来又发展出轮渡的交通方式,同时客户端代码的逻辑也变得更加复杂,对交通方式的实例类的调用也更频繁,原来只是调用 w.transfer(people),现在还需要执行载货、或者设备报告等等方式,这意味着每次引入一个新的交通方式类,就要保证这个新的类的引入不会导致客户端的代码不会出错。

  1. 新的类没有实现的方法,而客户端代码中在后面调用了
  2. 有些行为只有某一个产品才有
  3. 新的维护者用其他方式命名了该方法

为了规避这些问题,开发者对这些交通方式(产品)进行了抽象(抽象产品),任何新引入的交通方式必须继承这个抽象产品,并且实现这个抽象产品中的基本方法,于是这个类关系图变成:
在这里插入图片描述
而据此实现的新版本:

from abc import ABC, abstractmethod

class Product(ABC):
    @abstractmethod
    def 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 = 100
    w = Ship()
    w.transfer(people)

main()

在这种方式下,如果继承自 Product 的交通方式不实现其定义的抽象接口就无法实例化。进行到这里实际上关于工厂函数的四个角色,我们已经引出了两个,即下面标红的抽象产品具体产品,指代我们的交通方式具体交通方式

  1. Creator
  2. Concrete Creator
  3. Product
  4. Concrete Product

逐渐复杂的案例

让我们继续给这个上面的案例加多一些现实场景:

随着发展,每一种具体的交通方式都有了各自复杂多样的类型(具备不同属性的实例)。以前只有一辆24座的红色巴士,现在城市发展了,有50座红色巴士、50座绿色巴士、使用新能源的、使用汽油的…这样的情况在地铁和轮船上也是如此。
而针对运输一批市民的需求,也要根据实际情况不同选择合适的产品,就算是用巴士运载,也要选用合适座位数等属性的大巴车进行调度。

这体现在我们的代码中是怎样的呢?以 Bus 为例:

class Bus(Product):
    def __init__(self, seat, color, energy):
        self.color = color
        self.seat = seat
        self.energy = energy
        
    def transfer(self, people):
        print(f"move {people} mans to other place on Road")

def main():
    people = 100
    if people < 24:
        w = Bus(color="red", seat=24, energy="电")
    else:
        w = Bus(color="red", seat=120, energy="油")
    w.transfer(people)

main()

可以看到,客户端代码需要根据情景不同生成不同属性的实例,这还只是考虑了一个属性(人数)的逻辑,如果还需根据其他基本信息,那客户端代码就更复杂了,比如:

  1. 目的地不同选择电还是油?
  2. 同样是人数小于 24人时,生成轮船所需的配置参数和大巴的配置参数不同

这些问题如果都交给客户端去解决的话,那就会导致:

  1. 一旦引入新的产品就要修改一次代码
  2. 客户端需要负责实例的生成,高度的耦合

那么可不可以客户端只负责提出:

  1. 期望的产品(实例类型:大巴还是轮船)
  2. 问题的基本信息(人数、目的地…)即生成实例的基本参数

这样的背景下,我们引入创建者角色,由创建者根据不同的入参创建不同属性配置的实例

client code
Bus Creator
Specify Bus

而轮船也有其对应的轮船创建者、地铁对应其地铁创建者,这类创建者多了之后就衍生了规范,每一种创建者需要遵循某种基本规范:

Abstract Creator
Bus Creator
Ship Creator
Subway Creator

讲到这里,我们就引入了抽象创建者具体创建者, 到此工厂函数模式的四个角色就都到齐了。让我们把这新出现的两个角色也整合进代码中:

from abc import ABC, abstractmethod

class Product(ABC):
    @abstractmethod
    def transfer(self, people):
        ...

class Bus(Product):
    def __init__(self, seat, color, energy):
        self.color = color
        self.seat = seat
        self.energy = energy
    
    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")

class Creator(ABC):
    @abstractmethod
    def create_transportation(self):
        ...

class BusCreator(Creator):
    def __init__(self, people):
        self.expected_people = people
    
    def 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 = 10
    creator = 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:具体工厂

这四种角色,上面在案例中已经做了拆解就不做赘述。

为什么会用到工厂函数模式?

在上面的案例演变中,我觉得你应该得到这个问题的答案。如果没有,那么概括下:

  1. 如果这是一个简单的对象,不涉及复杂的实例创建过程,那么不需要应用这个模式,反过来讲,使用工厂函数模式是为了解决在客户端代码需要根据具体场景创建具体的复杂实例的问题。

工厂函数模式和抽象工厂模式有什么关系?

到点吃饭了,这个问题暂时先不展开论述了,其实你仔细看上面的代码可以发现有部分抽象工厂模式的思想。

  • 虽然代码示例中没有直接展示出抽象工厂模式的完整结构,但可以看到 Product 是一个抽象类,定义了产品的抽象接口 transfer(),而具体的产品类 BusSubwayShip 实现了这个接口。
  • Creator 类可以被视为一个抽象工厂,定义了一个创建产品的抽象方法 create_transportation(),而 BusCreatorShipCreator 则是具体工厂,实现了 Creator 接口并负责创建具体产品。

关系和区别

  • 关系:工厂函数模式可以被看作是抽象工厂模式的一种简化形式,特别是在只需创建单一种类的情况下。它们都用于将对象的创建与使用代码分离,提高系统的可扩展性和灵活性。
  • 区别:主要区别在于抽象工厂模式更加抽象和灵活,能够创建一组相关对象(产品家族),而工厂函数模式通常只涉及单一对象类型的创建。抽象工厂模式还涉及多个抽象类或接口的定义和实现,更适用于需要创建多个相关对象的场景。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Vue3学习笔记第一天

MVVM Vue是一种用于构建用户界面的JavaScript框架。MVVM 是Vue采用的一种软件架构模式&#xff0c;用于构建交互式的用户界面。它的全称是 Model-View-ViewModel&#xff0c;这三个部分分别代表了应用程序的不同层次和角色&#xff1a; Model&#xff08;模型&#xff09;&…

【ARM】v8架构programmer guide(3)_ARMv8的寄存器

目录 4.ARMv8 registers 4.1 AArch64 特殊寄存器 4.1.1 Zero register 4.1.2 Stack pointer &#xff08;SP) 4.1.3 Program Counter &#xff08;PC) 4.1.4 Exception Link Register(ELR) 4.1.5 Saved Process Status Register &#xff08;SPSR&#xff09; 4.2 Proc…

性能测试基础概念

前言&#x1f440;~ 上一章我们介绍了单元测试Junit的使用&#xff0c;今天我们来讲解一下性能测试的一些基础概念为后面我们进行性能测试做铺垫 什么是性能测试&#xff1f; 性能测试和功能测试有什么区别&#xff1f; 影响一个软件性能因素有哪些&#xff1f; 为什么要进…

循环神经网络和自然语言处理一

目录 一.分词 1.分词工具 2.分词的方法 3.N-gram表示方法 二.向量化 1.one-hot编码 2.word embedding 3.word embedding API 4.数据形状改变 既然是自然语言&#xff0c;那么就有字&#xff0c;词&#xff0c;句了 一.分词 1.分词工具 tokenization&#xff0c;jie…

Outlook Pst文件大小最大多大?如何分开缩减?

簡介 預設情況下&#xff0c;personal Folders (.pst) 和離線 Outlook Data File (.ost) 檔案在 Microsoft Outlook 2010 和 Outlook 2013 中為 Unicode 格式。 .pst 和 .ost 檔案的整體大小有 50 GB 的預先設定限制。 此限制大於 2007 和 Outlook 2003 Outlook Unicode .pst …

零基础5分钟上手亚马逊云科技AWS核心云开发/云架构知识 - 成本分析篇

简介&#xff1a; 欢迎来到小李哥全新亚马逊云科技AWS云计算知识学习系列&#xff0c;适用于任何无云计算或者亚马逊云科技技术背景的开发者&#xff0c;让大家零基础5分钟通过这篇文章就能完全学会亚马逊云科技一个经典的服务开发架构方案。 我将每天介绍一个基于亚马逊云科…

数据结构 - 相邻节点迭代器

&#x1f49d;&#x1f49d;&#x1f49d;首先&#xff0c;欢迎各位来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里不仅可以有所收获&#xff0c;同时也能感受到一份轻松欢乐的氛围&#xff0c;祝你生活愉快&#xff01; 文章目录 引言一、相邻节…

nuxt3实战:完整的 nuxt3 + vue3 项目创建与useFetch请求封装

一. 安装 pnpm dlx nuxilatest init <project-name>// ornpx nuxilatest init <project-name>如遇到报错 手动安装&#xff1a; 浏览器访问报错https请求地址&#xff1a; 点击tar(项目初始文件的下载地址)对应地址,下载starter-3.tar.gz 包到本地 本地创建项…

【Android】使用网络技术——WebView的用法、http协议、OKHttp、解析XML、JSON格式数据笔记整理

WebView的用法 新建一个WebView项目 修改activity_main中的代码&#xff1a; <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.android.com/apk/res/android"android:id"id/main"and…

STM32F1之SysTick系统定时器详细解析

目录 1. 简介 2. SysTick功能框图 3. SysTick寄存器 3.1 SysTick控制及状态寄存器 3.2 SysTick重装载数值寄存器 3.3 SysTick当前数值寄存器 3.4 SysTick校准数值寄存器 4. SysTick定时时间计算 5. SysTick寄存器结构体 6. 写一个us级延时函数 7. 写一个…

240806-RHEL 无法通过 ssh username@ip 远程连接,报错:Connection closed by ip port 22

A. 原因排查 遇到这个错误通常意味着 SSH 服务可能在目标主机上没有正常运行&#xff0c;或有防火墙/网络配置问题。以下是一些排查步骤&#xff1a; 检查 SSH 服务状态&#xff1a; 确认 SSH 服务是否正在目标主机上运行。 sudo systemctl status sshd重启 SSH 服务&#xff…

探索 Python 异步通信的奥秘:WebSockets 库的神奇之旅

文章目录 探索 Python 异步通信的奥秘&#xff1a;WebSockets 库的神奇之旅背景&#xff1a;为何选择 WebSockets&#xff1f;什么是 websockets 库&#xff1f;安装 websockets 库5个简单的库函数使用方法场景应用示例常见问题与解决方案总结 探索 Python 异步通信的奥秘&…

用Manim实现三维坐标系的绘制

1.ThreeDAxes 函数 ThreeDAxes是 Manim 中用于创建三维坐标系的类。在manim中常用的三位坐标绘制函数是&#xff1a; class ThreeDAxes(x_range(-6, 6, 1), y_range(-5, 5, 1), z_range(-4, 4, 1), x_length10.5, y_length10.5, z_length6.5, z_axis_configNone, z_normala…

数据仓库怎么建设?一文详解数仓的建设过程!

随着信息技术的飞速发展&#xff0c;企业不仅需要存储和管理海量数据&#xff0c;更迫切需要从这些数据中提取有价值的信息&#xff0c;以支持复杂的决策制定过程。数据仓库不仅是存储数据的场所&#xff0c;更是支持复杂查询、报告和数据分析的强有力工具&#xff0c;其建设已…

JavaScript异步简介|Promise快速入门

异步&#xff08;Asynchronous, async&#xff09;是与同步&#xff08;Synchronous, sync&#xff09;相对的概念。 异步 JavaScript 简介 异步编程技术使你的程序可以在执行一个可能长期运行的任务的同时继续对其他事件做出反应而不必等待任务完成。与此同时&#xff0c;你…

Linux工具|运维工具rename常用命令详解

&#x1f4eb; 作者简介&#xff1a;「六月暴雪飞梨花」&#xff0c;专注于研究Java&#xff0c;就职于科技型公司后端工程师 &#x1f3c6; 近期荣誉&#xff1a;华为云云享专家、阿里云专家博主、腾讯云优秀创作者、ACDU成员 &#x1f525; 三连支持&#xff1a;欢迎 ❤️关注…

【vulnhub】Wakanda :1靶机

靶机安装 下载地址&#xff1a;https://download.vulnhub.com/wakanda/wakanda-1.ova 运行环境&#xff1a;Virtual Box 信息收集 靶机IP扫描 netdiscover -i eth0 -r 192.168.7.0/24 端口扫描 nmap -A 192.168.7.243 -p- 80端口开启了http服务&#xff0c;在3333端口开启…

案例研究丨盛泰光电携手DataEase实现数据驱动智能制造

盛泰光电科技股份有限公司&#xff08;以下简称为“盛泰光电”&#xff09;是中国第一批摄像头模组制造企业。自成立至今&#xff0c;一直专注于手机摄像头模组的研发、制造、销售与服务&#xff0c;并向非手机包括笔记本、车载、医疗、AIoT等领域延伸&#xff0c;形成以手机摄…

PHP + Laravel + RabbitMQ + Redis 实现消息队列 (二) 消费队列在RabbitMQ和redis中的简单使用

最简单的队列功能 RabbitMQ和消息传递通常会使用一些术语&#xff1a; 生产者&#xff08;Producer&#xff09;意味着发送消息。一个发送消息的程序称为生产者。队列&#xff08;Queue&#xff09;尽管消息通过RabbitMQ和您的应用程序流动&#xff0c;但它们只能存储在队列中…

数据结构(01):数据结构概述(基本术语、逻辑结构和物理结构)

1、数据结构概述 (1)基本术语 A.数据元素 具有一定意义的基本单位。如人类的数据元素是人&#xff08;张三、李四等&#xff09;。 B.数据项 可以看作是数据元素的属性。如人的属性&#xff08;姓名、年龄、身高等&#xff09; C.数据对象 性质相同的数据元素的集合。如某一栋…