Python设计模式 - 建造者模式

定义

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

结构

在这里插入图片描述

  • 抽象建造者(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中,这样做更符合单一职责原则。

参考

《设计模式的艺术》

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

sparkTTS window 安装

SparkTTS 的简介 Spark-TTS是一种基于SpardAudio团队提出的 BiCodec 构建的新系统,BiCodec 是一种单流语音编解码器,可将语音策略性地分解为两种互补的标记类型:用于语言内容的低比特率语义标记和用于说话者特定属性的固定长度全局标记。这种…

高效微调算法 (Parameter-Efficient Fine-tuning, PEFT) 详解

引言 随着预训练语言模型 (Pre-trained Language Models, PLMs) 规模的持续膨胀,全参数微调 (Full Fine-tuning) 模式的局限性日益凸显。 全参数微调在下游任务上取得了显著的性能提升,但其高昂的计算和存储成本,以及为每个下游任务维护完整…

第十五届蓝桥杯大学B组(握手问题、小球反弹、好数)

一、握手问题 思路1&#xff1a; 1)先让所有人相互握手 第一个人49次 第二个人48次 第五十个人0次 共计01249 2)减去7个没握手的 016 #include<stdio.h> int main() {int a 50*49/2 - 7*6/2;printf("%d\n",a);return 0; } 运行结果&#xf…

若依框架-给sys_user表添加新字段并获取当前登录用户的该字段值

目录 添加字段 修改SysUser类 修改SysUserMapper.xml 修改user.js 前端获取字段值 添加字段 若依框架的sys_user表是没有age字段的&#xff0c;但由于业务需求&#xff0c;我需要新添加一个age字段&#xff1a; 修改SysUser类 添加age字段后&#xff0c;要在SysUser类 …

基于langchain+llama2的本地私有大语言模型实战

Langchain功能 LangChian 作为一个大语言模型&#xff08;LLM, Large Language Model&#xff09;开发框架&#xff0c;是 LLM 应用架构的重要一环。借助 LangChain&#xff0c;我们可以创建各种应用程序&#xff0c;包括聊天机器人和智能问答工具。 AI模型&#xff1a;包含各…

再聊 Flutter Riverpod ,注解模式下的 Riverpod 有什么特别之处,还有发展方向

三年前我们通过 《Flutter Riverpod 全面深入解析》 深入理解了 riverpod 的内部实现&#xff0c;而时隔三年之后&#xff0c;如今Riverpod 的主流模式已经是注解&#xff0c;那今天就让我们来聊聊 riverpod 的注解有什么特殊之处。 前言 在此之前&#xff0c;我们需要先回忆…

uniapp+Vue3 组件之间的传值方法

一、父子传值&#xff08;props / $emit 、ref / $refs&#xff09; 1、props / $emit 父组件通过 props 向子组件传递数据&#xff0c;子组件通过 $emit 触发事件向父组件传递数据。 父组件&#xff1a; // 父组件中<template><view class"container">…

Kafka×DeepSeek:智能决策破取经八十一难!

《西游记》的故事中&#xff0c;唐僧师徒四人历经九九八十一难&#xff0c;从东土大唐前往西天取经。一路上&#xff0c;火焰山酷热难耐、通天河水位忽高忽低、妖怪神出鬼没…… 现在&#xff0c;唐僧师徒取经路上的种种难题&#xff0c;在KafkaDeepSeek双引擎加持下有了全新解…

C# 委托使用详解

总目录 前言 在C#中&#xff0c;委托&#xff08;Delegate&#xff09; 是一种类型安全的函数指针机制&#xff0c;它允许我们将方法作为参数传递给其他方法&#xff0c;或者将方法存储在变量中。委托在 C# 中有广泛的应用&#xff0c;特别是在事件处理、异步编程和回调机制中…

axure11安装教程包含下载、安装、汉化、授权(附安装包)图文详细教程

文章目录 前言一、axure11安装包下载二、axure11安装教程1.启动安装程序2.安装向导界面3.安装协议协议页面2.选择安装位置3.开始安装4.完成安装 三、axure11汉化教程1.axure11汉化包2.axure11汉化设置 四、axure11授权教程1.打开axure112.设置使用方式3.输入许可证号4.axure11安…

如何使用Opentelemetry+jaeger对Go与Java项目实现分布式链路追踪

本文介绍![如何使用Opentelemetryjaeger实现分布式链路追踪] 关于opentelemetry的介绍可以看下面的文章 https://blog.csdn.net/qq_62368250/article/details/143516314本文中相关图片以及源代码地址 https://github.com/wuchenyanghaoshuai/others/blob/main/step39/README.…

【数据分享】2001-2024年我国逐年植被净初级生产力(NPP)数据

植被净初级生产力&#xff08;Net Primary Productivity&#xff0c;NPP&#xff09;是生态学中的一个重要概念&#xff0c;表示单位面积植被在特定时间内吸收的净光合有机物&#xff0c;是衡量生态系统中植物通过光合作用所产生的有机物质减去植物呼吸作用消耗的有机物质的量&…

靶场(七)---靶场精做小白思考

启程&#xff1a; 先扫一遍全端口发现&#xff0c;有很多tcp端口全部被关闭了&#xff0c;于是我又去看看他们的udp端口&#xff0c;发现也是半死不活的样子&#xff0c;那没办法只能把udp端口当作备选方案&#xff08;其实这个udp什么用都没有就是关闭的状态不用关&#xff0…

Linux开发工具----vim

目录 Linux编辑器-vim使用 1. vim的基本概念 正常/普通/命令模式(Normal mode) 插入模式(Insert mode) 底行模式(last line mode) 2. vim的基本操作 3. vim正常模式命令集 4. vim底行模式命令集 5. vim操作总结 (本篇文章相当于vim常用命令字典) Linux编辑器-vim使用 我们先来看…

【设计模式】设计模式的分类与组织

文章目录 前言一、设计模式的分类1. 目的准则2. 范围准则 二、设计模式的细分1.创建型模式的细分2.结构型模式的细分3.行为型模式的细分 三、设计模式的关联结论 前言 在软件开发中&#xff0c;设计模式是一种解决特定问题的最佳实践。由于设计模式种类繁多&#xff0c;理解它…

Vue3实战学习(Element-Plus常用组件的使用(输入框、下拉框、单选框多选框、el-image图片))(上)(5)

目录 一、Vue3工程环境配置、项目基础脚手架搭建、Vue3基础语法、Vue3集成Element-Plus的详细教程。(博客链接如下) 二、Element-Plus常用组件使用。 &#xff08;1&#xff09;el-input。(input输入框) <1>正常状态的el-input。 <2>el-input的disable状态。 <3…

AttributeError: module ‘backend_interagg‘ has no attribute ‘FigureCanvas‘

AttributeError: module backend_interagg has no attribute FigureCanvas 这个错误通常是由于 Matplotlib 的后端配置问题引起的。具体来说&#xff0c;Matplotlib 在尝试加载某个后端时&#xff0c;发现该后端模块中缺少必要的属性&#xff08;如 FigureCanvas&#xff09;&a…

网络安全基础与应用习题 网络安全基础答案

1.列出并简要给出SSH的定义。 正确答案&#xff1a; 答&#xff1a;6.10传输层协议&#xff1a;提供服务器身份验证、数据保密性和数据完整性&#xff0c;并具有前向保密性&#xff08;即&#xff0c;如果在一个会话期间密钥被破坏&#xff0c;则知识不会影响早期会话的安全性&…

文件上传复现

1、什么是文件上传漏洞&#xff1f; 答&#xff1a;文件上传漏洞是指攻击者通过上传恶意文件到服务器、从而执行任意代码、获取系统权限或者破坏系统安全的漏洞、常见于允许用户上传文件的Web应用程序中。 2. 文件上传漏洞形成原因 未验证文件类型&#xff1a;未对上传文件的…

【网络安全工程】任务12:网络安全设备

目录 一、防火墙​ 1、作用​ 2、配置方式​ 3、存在的漏洞​ 二、入侵检测系统&#xff08;IDS&#xff09;和入侵防御系统&#xff08;IPS&#xff09;​ 1、作用​ 2、配置方式​ 3、存在的漏洞​ 三、防病毒网关​ ​1、作用​ 2、配置方式​ 3、存在的漏洞​ …