【设计模式】Python 设计模式之建造者模式(Builder Pattern)详解

news2024/10/18 16:43:56

Python 设计模式之建造者模式(Builder Pattern)详解

在软件开发中,创建复杂对象往往需要多个步骤,而这些步骤之间的顺序、配置可能有多种变化。为了解决这个问题,建造者模式(Builder Pattern)应运而生。它可以将对象的构建过程与对象的表示分离,使得同样的构建过程可以创建不同的表示。

本文将详细介绍Python中的建造者模式,探讨其原理、适用场景、具体实现方式及优化方向,帮助开发者更好地理解和运用这一设计模式。

什么是建造者模式?

建造者模式是一种创建型设计模式,用于将一个复杂对象的构建过程分解为多个步骤,并通过一个**指挥者(Director)**来按照这些步骤来构造对象。建造者模式的核心思想是将对象的构造与对象的表示(如何创建对象)分离开来,使得同样的构建过程可以生成不同类型或配置的对象。

建造者模式的角色

建造者模式主要包括以下四个角色:

  1. 产品(Product):表示需要构建的复杂对象。
  2. 建造者(Builder):定义创建产品的抽象接口,包含构造产品不同部分的方法。
  3. 具体建造者(ConcreteBuilder):实现 Builder 接口,完成复杂对象各个部分的实际构造工作。
  4. 指挥者(Director):负责管理 Builder,按照顺序调用 Builder 中的方法来一步步构建产品对象。

建造者模式的应用场景

建造者模式适用于以下几种场景:

  1. 构建复杂对象:如果对象的构造步骤非常复杂(需要许多配置项),且构建过程中步骤固定但顺序不同或部分步骤可选,则适合用建造者模式。
  2. 隔离复杂对象的创建和使用:在某些情况下,我们希望隔离复杂对象的创建逻辑和使用逻辑,使用建造者模式可以让用户专注于如何使用对象,而不需要关心对象的构造过程。
  3. 多种配置对象:当一个类的实例具有不同的配置方式时,建造者模式可以帮助简化对象的创建。

典型的应用场景包括:构建复杂的UI界面、文档生成器、汽车生产、餐厅点餐系统等。

建造者模式的结构

建造者模式的基本结构可以用UML类图表示如下:

+-----------------+        +-------------------+
|     Director    | <----> |      Builder       |
+-----------------+        +-------------------+
                             | build_part1()    |
                             | build_part2()    |
                             +-------------------+
                                        |
                                        V
                              +-------------------+
                              | ConcreteBuilder   |
                              +-------------------+
                              | build_part1()     |
                              | build_part2()     |
                              +-------------------+
                                        |
                                        V
                              +-------------------+
                              |      Product      |
                              +-------------------+

在这个结构中,Director 负责指挥 Builder 进行产品的创建,Builder 定义了构建产品的各个步骤,而 ConcreteBuilder 则负责实现这些步骤并生成最终的产品。

Python 实现建造者模式

接下来,我们通过一个具体的例子来实现建造者模式。假设我们要构建一个复杂的对象——房子(House),房子由不同的部分组成,如地基、墙壁、屋顶等。不同的房子类型可能有不同的部件配置(比如豪宅和普通房子)。

1. 定义产品类(Product)

房子是我们需要构建的复杂对象,因此我们首先定义一个 House 类。

class House:
    def __init__(self):
        self.foundation = None
        self.structure = None
        self.roof = None
        self.interior = None

    def __str__(self):
        return f"Foundation: {self.foundation}, Structure: {self.structure}, Roof: {self.roof}, Interior: {self.interior}"

2. 定义建造者接口(Builder)

接下来,定义 Builder 抽象类,它包含构建房子不同部分的抽象方法。

from abc import ABC, abstractmethod

class HouseBuilder(ABC):
    @abstractmethod
    def build_foundation(self):
        pass

    @abstractmethod
    def build_structure(self):
        pass

    @abstractmethod
    def build_roof(self):
        pass

    @abstractmethod
    def build_interior(self):
        pass

    @abstractmethod
    def get_house(self):
        pass

3. 实现具体建造者类(ConcreteBuilder)

我们需要具体的建造者类来构建不同类型的房子。比如,下面是豪宅(MansionBuilder)和普通房子(StandardHouseBuilder)的实现:

class MansionBuilder(HouseBuilder):
    def __init__(self):
        self.house = House()

    def build_foundation(self):
        self.house.foundation = "Concrete, reinforced with steel"

    def build_structure(self):
        self.house.structure = "Steel and Glass"

    def build_roof(self):
        self.house.roof = "Solar panel roof"

    def build_interior(self):
        self.house.interior = "Luxury interior with marble floors"

    def get_house(self):
        return self.house

class StandardHouseBuilder(HouseBuilder):
    def __init__(self):
        self.house = House()

    def build_foundation(self):
        self.house.foundation = "Concrete"

    def build_structure(self):
        self.house.structure = "Wood and brick"

    def build_roof(self):
        self.house.roof = "Shingle roof"

    def build_interior(self):
        self.house.interior = "Basic interior with wooden floors"

    def get_house(self):
        return self.house

4. 定义指挥者类(Director)

Director 类负责控制构建过程,它接受一个 Builder 对象,并调用构建步骤来生成最终的产品。

class HouseDirector:
    def __init__(self, builder):
        self.builder = builder

    def construct_house(self):
        self.builder.build_foundation()
        self.builder.build_structure()
        self.builder.build_roof()
        self.builder.build_interior()
        return self.builder.get_house()

5. 测试建造者模式

接下来,创建 HouseDirector 并传入不同的 Builder 来构建不同的房子。

# 测试建造者模式
mansion_builder = MansionBuilder()
director = HouseDirector(mansion_builder)
mansion = director.construct_house()
print("Mansion:", mansion)

standard_builder = StandardHouseBuilder()
director = HouseDirector(standard_builder)
standard_house = director.construct_house()
print("Standard House:", standard_house)

输出结果:

Mansion: Foundation: Concrete, reinforced with steel, Structure: Steel and Glass, Roof: Solar panel roof, Interior: Luxury interior with marble floors
Standard House: Foundation: Concrete, Structure: Wood and brick, Roof: Shingle roof, Interior: Basic interior with wooden floors

改进建造者模式:链式调用

我们可以对建造者模式进行改进,使得它支持链式调用,这样可以使代码更加简洁、流畅。

class FluentHouseBuilder:
    def __init__(self):
        self.house = House()

    def build_foundation(self, foundation):
        self.house.foundation = foundation
        return self

    def build_structure(self, structure):
        self.house.structure = structure
        return self

    def build_roof(self, roof):
        self.house.roof = roof
        return self

    def build_interior(self, interior):
        self.house.interior = interior
        return self

    def get_house(self):
        return self.house

使用链式调用构建房子:

builder = FluentHouseBuilder()
house = (builder.build_foundation("Concrete")
         .build_structure("Wood and brick")
         .build_roof("Shingle roof")
         .build_interior("Basic interior with wooden floors")
         .get_house())

print(house)

这种方式让构建过程更加简洁,特别是在构建过程中的步骤较多时,链式调用可以减少代码冗余。

总结

建造者模式是一种非常适合构建复杂对象的设计模式,尤其是在对象的创建步骤较多、且顺序或配置变化较大的情况下。本文通过定义产品、抽象建造者、具体建造者以及指挥者,详细介绍了如何在Python中实现建造者模式。

建造者模式的好处在于它能够将对象的创建过程与其表示分离,便于扩展和维护。同时,我们还展示了如何通过链式调用的方式优化建造者模式,使代码更加简洁和可读。

希望这篇博客能帮助你更好地理解建造者模式,并能够

灵活应用到实际项目中。如果你有任何问题或建议,欢迎在评论区讨论!

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

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

相关文章

Qt(10.11)

数据表 源代码&#xff1a; #include "widget.h" #include "ui_widget.h" #include<QMessageBox>//消息对话框 #include<QDebug> #include<QSqlRecord> Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui-&g…

十、pico+Unity交互开发教程——射线抓取与更多交互功能

一、回顾与引入 回顾上一篇直接抓取的教程&#xff0c;VR交互一般需要可交互的对象&#xff08;Interactable&#xff09;和发起交互的对象&#xff08;Interactor&#xff09;。直接抓取和射线抓取的可交互对象无区别&#xff0c;可参考上一篇教程设置组件。两者区别在于发起…

【Vercel】Vercel静态部署踩坑

背景 在现代的软件开发中&#xff0c;自动化部署是一个不可或缺的环节。Vercel作为一个流行的前端部署平台&#xff0c;提供了与GitHub的无缝集成&#xff0c;使得开发者能够在每次提交代码后自动触发部署流程。然而&#xff0c;自动化部署过程中可能会遇到一些挑战&#xff0…

性能测试:流量回放工具-GoReplay!结合一款无需CA证书即可抓取HTTPS明文的工具,简直无敌

性能测试&#xff1a;流量回放工具-GoReplay&#xff01;结合一款无需CA证书即可抓取HTTPS明文的工具&#xff0c;简直无敌。 GoReplay 是一个开源网络监控工具&#xff0c;可以将实时 HTTP 流量捕获并重放到测试环境。 应用成熟的过程中&#xff0c;测试所需的工作量往往会成…

学习干货小白女友看完这篇文章后,面试工作和护网蓝队初级竟然秒通过!

小白女友看完这篇文章后&#xff0c;面试工作和护网蓝队初级竟然秒通过&#xff01; 前言&#xff1a;本文中涉及到的相关技术或工具仅限技术研究与讨论&#xff0c;严禁用于非法用途&#xff0c;否则产生的一切后果自行承担&#xff0c;如有侵权请联系。 还在学怎么挖通用漏…

【Linux】<互斥量>解决<抢票问题>——【多线程竞争问题】

前言 大家好吖&#xff0c;欢迎来到 YY 滴Linux系列 &#xff0c;热烈欢迎&#xff01; 本章主要内容面向接触过C的老铁 主要内容含&#xff1a; 欢迎订阅 YY滴C专栏&#xff01;更多干货持续更新&#xff01;以下是传送门&#xff01; YY的《C》专栏YY的《C11》专栏YY的《Lin…

Axure树形菜单展开与折叠

亲爱的小伙伴&#xff0c;在您浏览之前&#xff0c;烦请关注一下&#xff0c;在此深表感谢&#xff01; 课程主题&#xff1a;Axure树形菜单展开与折叠 主要内容&#xff1a;树形菜单制作——层级关系——隐藏与显示——值的变化——多层交互 应用场景&#xff1a;关系树、菜…

老机MicroServer Gen8再玩 OCP万兆光口+IT直通

手上有一台放了很久的GEN8微型服务器&#xff0c;放了很多年&#xff0c;具体什么时候买的我居然已经记不清了 只记得开始装修的时候搬家出去就没用了&#xff0c;结果搬出去有了第1个孩子&#xff0c;孩子小的时候也没时间折腾&#xff0c;等孩子大一点的时候&#xff0c;又有…

MongoDB查询操作

&#x1f337;启动mongo &#x1f388;启动mongo shell &#xff08;1&#xff09;在指定目录下创建mongodb文件夹、其子文件夹data、log以及文件mongodb.log cd /home/ubuntu mkdir -p mongodb/data mkdir -p mongodb/log touch mongodb/log/mongodb.log(2)先执行mongodb命…

《计算机视觉》—— 疲劳检测

文章目录 一、疲劳检测实现的思想二、代码实现 一、疲劳检测实现的思想 了解以下几篇文章有助于了解疲劳检测的方法 基于dlib库的人脸检测 https://blog.csdn.net/weixin_73504499/article/details/142977202?spm1001.2014.3001.5501 基于dlib库的人脸关键点定位 https://blo…

基于开源Jetlinks物联网平台协议包-MQTT自定义主题数据的自动回复

目录 1.根据需要自动回复某些主题 2.调用doReply方法进行自动回复 1.根据需要自动回复某些主题 根据主题判断&#xff0c;哪些主题是需要自动回复的&#xff0c;比如设备登录&#xff0c;需要自动回复。 2.调用doReply方法进行自动回复&#xff08;代码不一定全部正确&#xf…

第 5 章:vuex

1. 理解 vuex vuex 是什么&#xff1a; 概念&#xff1a;专门在 Vue 中实现集中式状态&#xff08;数据&#xff09;管理的一个 Vue 插件&#xff0c;对 vue 应用中多个组件的共享状态进行集中式的管理&#xff08;读/写&#xff09;&#xff0c;也是一种组件间通信的方式&am…

安乃达:用CRM构建从销售到管理到售后的全链路数字化运营平台

安乃达驱动技术(上海)股份有限公司((简称&#xff1a;“安乃达”,股票代码为&#xff1a;“603350”))自2011年以来&#xff0c;公司通过多年的研发与积累现有直驱轮毂电机、减速轮毂电机和中置电机三大系列产品&#xff0c;并具备与电机相匹配的控制器、传感器、仪表等电驱动成…

springboot旧物置换网站

作者&#xff1a;计算机学长阿伟 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、ElementUI等&#xff0c;“文末源码”。 系统展示 【2024最新】基于JavaSpringBootVueMySQL的&#xff0c;前后端分离。 开发语言&#xff1a;Java数据库&#xff1a;MySQL技术&#xff1a;…

ThinkPHP 3.2 + Nginx 页面404问题

学习公司Callout项目时&#xff0c;发现公司项目所使用的TP版本是3.2&#xff0c;所以才可以使用例如&#xff0c;C,M,A等方法 因此我用phpEnv搭建了一个项目&#xff0c;域名为thinkphp&#xff0c;所选根目录如下 我打开网页&#xff0c;访问 thinkphp/ 和 thinkphp/index.p…

ROS 的 urdf 中 link 和 joint 的子标签中 origin 的含义

主要参考文章——主要文章&#xff0c;官方关于urdf的介绍和官方文档的翻译解析 link标签里面的origin含义 link标签里面有三个主要的子标签&#xff0c;分别是visual——连杆的外观和坐标系&#xff0c;collisoin——连杆的碰撞属性和inertial——连杆的惯性设置 首先&…

C++ | AVL树

前言 本篇博客讲解c中数据结构AVL树&#xff0c;看这篇博客之前请先去看&#xff1a;C | 二叉搜索树-CSDN博客 &#x1f493; 个人主页&#xff1a;普通young man-CSDN博客 ⏩ 文章专栏&#xff1a;C_普通young man的博客-CSDN博客 ⏩ 本人giee: 普通小青年 (pu-tong-young-m…

2024最新:零基础到精通的大模型AI产品经理全学习路线

随着人工智能技术的发展&#xff0c;尤其是大模型&#xff08;Large Model&#xff09;的兴起&#xff0c;越来越多的企业开始重视这一领域的投入。作为大模型产品经理&#xff0c;你需要具备一系列跨学科的知识和技能&#xff0c;以便有效地推动产品的开发、优化和市场化。以下…

第51期 C语言实现中断<一>

Q&#xff1a;怎样理解用C语言实现中断的过程呢&#xff1f; A&#xff1a;以下是一段使用C语言实现中断的主程序&#xff0c;和汇编语言实现中断一样也使用了定时器中断和按键中断。执行该主程序会在DE2-115的红色LED上显示流水灯&#xff0c;按下KEY1可以改变流水灯移动的…

FreeRTOS - 任务通知

1. 任务通知 所谓"任务通知"&#xff0c;你可以反过来读"通知任务"。 我们使用队列、信号量、事件组等等方法时&#xff0c;并不知道对方是谁。使用任务通知时&#xff0c;可以明确指定&#xff1a;通知哪个任务。 使用队列、信号量、事件组时&#xff…