Python设计模式 - 建造者模式

news2025/3/14 21:18:12

定义

建造者模式是一种创建型设计模式,主要用于构建包含多个组成部分的复杂对象。它将对象的构建过程与表示分离,使得同样的构建过程可以创建不同的对象表示。

结构

在这里插入图片描述

  • 抽象建造者(Builder):声明创建产品的各个部件的方法,以及一个获取产品对象的方法。
  • 具体建造者(ConcreteBuilder):实现抽象建造者的方法,这些方法用来构建产品的各个部件。
  • 产品(Product):被构建的复杂对象。
  • 指挥者(Director):负责安排产品各部件的建造顺序,调用抽象建造者的方法构建产品的各个部分,最后返回创建好的产品对象。
  • 客户端(Client):使用指挥者和建造者来构建产品。

应用场景

  1. 对象结构复杂,创建过程涉及多个步骤:当对象由多个组件组成,且这些组件的初始化顺序或组合方式较灵活时,建造者模式可以将创建过程拆分为多个步骤,使得对象的构建更加清晰。例如,构建 Computer 对象时,可以逐步配置 CPU、内存、硬盘等组件。
  2. 需要灵活配置对象:当对象的某些部分是可选的,或者可以根据不同需求定制时,建造者模式能够提供流畅的配置方式,而不影响对象的整体创建。例如,创建游戏角色时,可以选择不同的装备、技能组合,而无需修改底层实现。

优缺点

优点:

  1. 创建过程与表示分离:建造者模式将对象的创建过程和最终表示解耦,使得同样的构造过程可以创建不同的对象变体
  2. 封装复杂创建过程:将对象的创建流程封装在指挥者中,客户端无需关心具体的构造细节,而是通过指挥者直接获取完整的对象,从而简化调用逻辑。
  3. 提高代码的可扩展性:新增产品时,只需扩展新的具体建造者,而无需修改现有代码,符合开闭原则。

缺点:

  1. 可能导致冗余代码:对于每种不同的产品类型,都需要创建对应的建造者类,这在产品种类较多但构建逻辑相似的情况下,可能导致大量重复代码,不如使用抽象工厂模式更合适。

代码示例

from abc import ABC, abstractmethod


class Computer:
    """ 计算机产品 """

    def __init__(self, brand):
        self.brand = brand
        self.cpu = None
        self.memory = None
        self.storage = None
        self.graphics_card = None  # 可选组件(独立显卡)

    def __str__(self):
        gpu_info = f", {self.graphics_card} GPU" if self.graphics_card else ", No GPU"
        return f"{self.brand} Computer with {self.cpu}, {self.memory} RAM, {self.storage} Storage{gpu_info}"


class ComputerBuilder(ABC):
    """ 计算机建造者基类(抽象类) """

    def __init__(self, brand):
        self.computer = Computer(brand)

    @abstractmethod
    def set_cpu(self):
        pass

    @abstractmethod
    def set_memory(self):
        pass

    @abstractmethod
    def set_storage(self):
        pass

    @abstractmethod
    def set_graphics_card(self):
        pass

    def build(self):
        return self.computer


class LenovoComputerBuilder(ComputerBuilder):
    """ 联想电脑建造者(默认带独立显卡) """

    def __init__(self):
        super().__init__("Lenovo")

    def set_cpu(self):
        self.computer.cpu = "Intel i7"
        return self

    def set_memory(self):
        self.computer.memory = "16GB"
        return self

    def set_storage(self):
        self.computer.storage = "512GB SSD"
        return self

    def set_graphics_card(self):
        self.computer.graphics_card = "NVIDIA RTX 4090"
        return self


class MacComputerBuilder(ComputerBuilder):
    """ Mac 电脑建造者(默认不带独立显卡) """

    def __init__(self):
        super().__init__("Mac")

    def set_cpu(self):
        self.computer.cpu = "Apple M2"
        return self

    def set_memory(self):
        self.computer.memory = "32GB"
        return self

    def set_storage(self):
        self.computer.storage = "1TB SSD"
        return self

    def set_graphics_card(self):
        return self


class Director:
    """ 指挥者,控制建造流程 """

    @staticmethod
    def construct_pc(builder: ComputerBuilder):
        """ 统一的构造过程 """
        return (builder.set_cpu()
                .set_memory()
                .set_storage()
                .set_graphics_card()
                .build())


# 客户端代码
lenovo_pc = Director.construct_pc(LenovoComputerBuilder())  # 联想默认带显卡
mac_pc = Director.construct_pc(MacComputerBuilder())  # Mac 默认无显卡

print(lenovo_pc)  # Lenovo Computer with Intel i7, 16GB RAM, 512GB SSD Storage, NVIDIA RTX 4090 GPU
print(mac_pc)  # Mac Computer with Apple M2, 32GB RAM, 1TB SSD Storage, No GPU

省略Director

在有些情况下,为了简化系统结构,可以将Director和抽象建造者Builder进行合并,在Builder中提供逐步构建复杂产品对象的construct()方法。

class ComputerBuilder(ABC):
    """ 计算机建造者基类(抽象类) """

    def __init__(self, brand):
        self.computer = Computer(brand)

    @abstractmethod
    def set_cpu(self):
        pass

    @abstractmethod
    def set_memory(self):
        pass

    @abstractmethod
    def set_storage(self):
        pass

    @abstractmethod
    def set_graphics_card(self):
        pass

    def construct(self):
        """ 统一的构造过程 """
        return (self.set_cpu()
                .set_memory()
                .set_storage()
                .set_graphics_card()
                .computer)
                
# 客户端代码:直接选择 Builder 并调用 construct()
lenovo_pc = LenovoComputerBuilder().construct()  # 联想默认带显卡
mac_pc = MacComputerBuilder().construct()  # Mac 默认无显卡

print(lenovo_pc)  # Lenovo Computer with Intel i7, 16GB RAM, 512GB SSD Storage, NVIDIA RTX 4090 GPU
print(mac_pc)  # Mac Computer with Apple M2, 32GB RAM, 1TB SSD Storage, No GPU

这种方式不影响系统的灵活性和可扩展性,同时还简化了系统结构,但是加重了抽象建造者类的职责。如果construct()方法较为复杂,待构建产品的组成部分较多,建议还是将construct()方法单独封装在Director中,这样做更符合单一职责原则。

参考

《设计模式的艺术》

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

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

相关文章

在 Ubuntu 上安装和配置 Docker 的完整指南

Docker 是一个开源的平台,旨在简化应用程序的开发、部署和运行。通过将应用程序及其依赖项打包到容器中,Docker 确保应用程序可以在任何环境中一致地运行。 目录 前言安装前的准备安装 Docker 步骤 1:更新包索引步骤 2:安装必要…

网络安全之数据加密(DES、AES、RSA、MD5)

刚到公司时,我的工作就是为app端提供相应的接口。之前app使用的是PHP接口,对数据加密方面做得比较少。到使用java接口时,老大开始让我们使用DES加密,进行数据传输,但是后来觉得DES是对称加密,密钥存在客户端…

基于SpringBoot的“校园周边美食探索及分享平台”的设计与实现(源码+数据库+文档+PPT)

基于SpringBoot的“校园周边美食探索及分享平台”的设计与实现(源码数据库文档PPT) 开发语言:Java 数据库:MySQL 技术:SpringBoot 工具:IDEA/Ecilpse、Navicat、Maven 系统展示 校园周边美食探索及分享平台结构图…

vscode关闭仓库后如何打开

vscode源代码管理->更改->代码 关闭仓库后如何打开。 关闭仓库操作 打开仓库操作 1.按下 Ctrl Shift P(Windows/Linux)或 Cmd Shift P(Mac)打开命令面板。 2.在命令面板中输入 Git: Open Repository,然后选…

DeepSeek-R1 论文阅读总结

1. QA问答(我的笔记) Q1: DeepSeek如何处理可读性问题? 通过构建冷启动数据(数千条长CoT数据)微调基础模型,结合多阶段训练流程(RL训练、拒绝采样生成SFT数据),并优化输…

Linux 》》Ubuntu 18 LTS 之后的版本 修改IP地址 主机名

进入目录 /etc/netplan 修改 50-cloud-init.yaml 》保存文件后,执行以下命令应用更改: sudo netplan apply 》》 DHCP模式 修改主机名 hostnamectl set-hostname xxxx 修改cloud.cfg 防止重启主机名还原 但测试下来 不修改, 重启 也不会还…

泰山派开发之—Ubuntu24.04下Linux开发环境搭建

简介 最近翻到了吃灰已久的泰山派,是刚出来的时候用优惠券买的,当时价格挺便宜的,最近给它翻出来了,打算试试做个项目。买的泰山派容量是2G16G,SOC芯片使用的是RK3566,搭载1TOP算力的NPU,并且具…

哈尔滨算力服务器托管推荐-青蛙云

哈尔滨年平均气温3.5摄氏度,有发展云计算和算力数据中心的天然优势 ,今天为哈尔滨算力服务器托管服务商:青蛙云,黑龙江经营17年的老牌IDC服务商。 先来了解下算力服务器: 算力服务器,尤其是那些用于运行人…

openharmony体验

openharmony5 去年已经出来了 如果以前做过android开发的,学起来不难,关键 1:环境 DevEco Studio 5.0.3 Beta2 https://developer.huawei.com/consumer/cn/deveco-studio/ win10_64bit CPU amd64(不是arm的) 2:安装 执行EXE 安装就行&#x…

[Ai 力扣题单] 数组基本操作篇 27/704/344/386

题单分类:DeepSeek刷力扣辅助题单 存留记录-CSDN博客 27 27. 移除元素 - 力扣(LeetCode) 这道题就一个点 1.数组在内存上连续 所以要么赋值覆盖,要么移动覆盖,但是它要求了前 k 个元素 所以只能移动覆盖 所以我有了如下思考过程: 3223 , 3举例 如果是…

Linux入门 全面整理终端 Bash、Vim 基础命令速记

Linux入门 2025 超详细全面整理 Bash、Vim 基础命令速记 刚面对高级感满满的 终端窗口是不是有点懵?于是乎,这份手册就是为你准备的高效学习指南!我把那些让人头大的系统设置、记不住的命令都整理成了对你更友好的格式,让你快速学…

AI智能代码疫苗技术,赋能数字化应用内生安全自免疫

“DevSecOps市占率持续领先,IAST探针覆盖率十倍增长,代码疫苗技术已成功帮助上千家行业用户成功抵御‘Log4j2.x’等重大未知漏洞的利用攻击。”子芽在腾讯专访中透露。 这是2021年悬镜安全交出的一张成绩单。悬镜安全是DevSecOps敏捷安全先行者&#xf…

《SQL性能优化指南:新手如何写出高效的数据库查询

新手程序员如何用三个月成为SQL高手?万字自学指南带你弯道超车 在数据为王的时代,掌握SQL已成为职场新人的必修课。你可能不知道,仅用三个月系统学习,一个零基础的小白就能完成从数据库萌新到SQL达人的蜕变。去年刚毕业的小王就是…

【TES817】基于XCZU19EG FPGA的高性能实时信号处理平台

板卡概述 TES817是一款基于ZU19EG FPGA的高性能实时信号处理平台,该平台采用1片高性能的FPGA:XCZU19EG-2FFVC1760I作为主处理器,FPGA的PL端外挂1组72位DDR4 SDRAM,用来实现超大容量数据缓存,FPGA的PS端外挂1组72位的D…

Python 进程与线程-分布式进程

目录 分布式进程 小结 分布式进程 在Thread和Process中,应当优选Process,因为Process更稳定,而且,Process可以分布到多台机器上,而Thread最多只能分布到同一台机器的多个CPU上。 Python的multiprocessing模块不但支…

初阶数据结构(C语言实现)——5.2 二叉树的顺序结构及堆的实现

1.二叉树的顺序结构及实现 1.1 二叉树的顺序结构 普通的二叉树是不适合用数组来存储的,因为可能会存在大量的空间浪费。而完全二叉树更适合使用顺序结构存储。现实中我们通常把堆(一种二叉树)使用顺序结构的数组来存储,需要注意的是这里的堆和操作系统…

ArcGIS Pro 车牌分区数据处理与地图制作全攻略

在大数据时代,地理信息系统(GIS)技术在各个领域都有着广泛的应用,而 ArcGIS Pro 作为一款功能强大的 GIS 软件,为数据处理和地图制作提供了丰富的工具和便捷的操作流程。 车牌数据作为一种重要的地理空间数据&#xf…

文件解析漏洞靶场通关合集

一、IIS解析漏洞 &#xff08;一&#xff09;iis6的目录解析漏洞(.asp目录中的所有文件都会被当做asp文件执行) 第一步&#xff1a;在网站根目录下创建了一个x.asp文件夹&#xff0c;并在文件夹中创建一个名为1.txt的文本文档 第二步&#xff1a;文本文档中输入<% now()%&…

塔能IVO-SCY智能机箱:点亮智慧城市的电力“智慧核芯”

在智慧城市建设的宏大征程中&#xff0c;稳定且智能的电力供应犹如坚固基石&#xff0c;支撑着各类设备高效、稳定地运行。塔能科技的IVO-SCY智能机箱&#xff0c;凭借其卓越的电源管理系统&#xff0c;当之无愧地成为了整个智慧城市电力保障体系中的“智慧心脏”&#xff0c;源…

【Oracle】19c数据库控制文件多路径配置

一、关闭数据库&#xff08;2个节点实例都要关闭&#xff09; srvctl stop database -d ora19c 二、多路径控制文件 打开其中一个节点到nomount状态 sqlplus / as sysdba startup nomount; [oracleora19c1:/home/oracle]$ rman target / RMAN> restore controlfile to…